#include <sys/param.h>
#include <sys/systm.h>
#include <sys/conf.h>
#include <sys/caprights.h>
#include <sys/disk.h>
#include <sys/eventhandler.h>
#include <sys/fcntl.h>
#include <sys/file.h>
#include <sys/kerneldump.h>
#include <sys/limits.h>
#include <sys/malloc.h>
#include <sys/namei.h>
#include <sys/priv.h>
#include <sys/proc.h>
#include <sys/stat.h>
#include <sys/sysctl.h>
#include <sys/vnode.h>
#include <machine/pcb.h>
#include <machine/vmparam.h>
static dumper_start_t vnode_dumper_start;
static dumper_t vnode_dump;
static dumper_hdr_t vnode_write_headers;
static struct sx livedump_sx;
SX_SYSINIT(livedump, &livedump_sx, "Livedump sx");
int
livedump_start(int fd, int flags, uint8_t compression)
{
#if MINIDUMP_PAGE_TRACKING == 1
struct file *fp;
struct vnode *vp;
int error;
error = priv_check(curthread, PRIV_KMEM_READ);
if (error != 0)
return (error);
if (flags != 0)
return (EINVAL);
error = getvnode(curthread, fd, &cap_write_rights, &fp);
if (error != 0)
return (error);
vp = fp->f_vnode;
if ((fp->f_flag & FWRITE) == 0) {
error = EBADF;
goto drop;
}
error = livedump_start_vnode(vp, flags, compression);
if (error != 0)
goto drop;
drop:
fdrop(fp, curthread);
return (error);
#else
return (EOPNOTSUPP);
#endif
}
int
livedump_start_vnode(struct vnode *vp, int flags, uint8_t compression)
{
#if MINIDUMP_PAGE_TRACKING == 1
struct dumperinfo di, *livedi;
struct diocskerneldump_arg kda;
void *rl_cookie;
int error;
bzero(&di, sizeof(di));
di.dumper_start = vnode_dumper_start;
di.dumper = vnode_dump;
di.dumper_hdr = vnode_write_headers;
di.blocksize = PAGE_SIZE;
di.maxiosize = MAXDUMPPGS * PAGE_SIZE;
bzero(&kda, sizeof(kda));
kda.kda_compression = compression;
error = dumper_create(&di, "livedump", &kda, &livedi);
if (error != 0)
return (error);
if (sx_try_xlock(&livedump_sx) == 0) {
dumper_destroy(livedi);
error = EBUSY;
return (error);
}
livedi->priv = vp;
rl_cookie = vn_rangelock_wlock(vp, 0, OFF_MAX);
vn_lock(vp, LK_EXCLUSIVE | LK_RETRY);
EVENTHANDLER_INVOKE(livedumper_start, &error);
if (error != 0)
goto out;
curthread->td_pflags2 |= TDP2_SAN_QUIET;
dump_savectx();
error = minidumpsys(livedi, true);
curthread->td_pflags2 &= ~TDP2_SAN_QUIET;
EVENTHANDLER_INVOKE(livedumper_finish);
out:
VOP_UNLOCK(vp);
vn_rangelock_unlock(vp, rl_cookie);
sx_xunlock(&livedump_sx);
dumper_destroy(livedi);
return (error);
#else
return (EOPNOTSUPP);
#endif
}
int
vnode_dumper_start(struct dumperinfo *di, void *key, uint32_t keysize)
{
di->dumpoff = 0;
KASSERT(keysize == 0, ("encryption not supported for livedumps"));
return (0);
}
int
vnode_dump(void *arg, void *virtual, off_t offset, size_t length)
{
struct vnode *vp;
int error = 0;
vp = arg;
MPASS(vp != NULL);
ASSERT_VOP_LOCKED(vp, __func__);
EVENTHANDLER_INVOKE(livedumper_dump, virtual, offset, length, &error);
if (error != 0)
return (error);
if (virtual == NULL)
return (0);
error = vn_rdwr(UIO_WRITE, vp, virtual, length, offset, UIO_SYSSPACE,
IO_NODELOCKED, curthread->td_ucred, NOCRED, NULL, curthread);
if (error != 0)
uprintf("%s: error writing livedump block at offset %jx: %d\n",
__func__, (uintmax_t)offset, error);
return (error);
}
int
vnode_write_headers(struct dumperinfo *di, struct kerneldumpheader *kdh)
{
struct vnode *vp;
int error;
off_t offset;
vp = di->priv;
MPASS(vp != NULL);
ASSERT_VOP_LOCKED(vp, __func__);
offset = roundup2(di->dumpoff, di->blocksize);
error = vn_rdwr(UIO_WRITE, vp, kdh, sizeof(*kdh), offset,
UIO_SYSSPACE, IO_NODELOCKED, curthread->td_ucred, NOCRED, NULL,
curthread);
if (error != 0)
uprintf("%s: error writing livedump header: %d\n", __func__,
error);
return (error);
}