示例#1
0
/*
 * PM notifier callback for kdump
 */
static int machine_kdump_pm_cb(struct notifier_block *nb, unsigned long action,
			       void *ptr)
{
	switch (action) {
	case PM_SUSPEND_PREPARE:
	case PM_HIBERNATION_PREPARE:
		if (crashk_res.start)
			crash_map_reserved_pages();
		break;
	case PM_POST_SUSPEND:
	case PM_POST_HIBERNATION:
		if (crashk_res.start)
			crash_unmap_reserved_pages();
		break;
	default:
		return NOTIFY_DONE;
	}
	return NOTIFY_OK;
}
示例#2
0
SYSCALL_DEFINE4(kexec_load, unsigned long, entry, unsigned long, nr_segments,
                struct kexec_segment __user *, segments, unsigned long, flags)
{
    struct kimage **dest_image, *image;
    int result;

    /* We only trust the superuser with rebooting the system. */
    if (!capable(CAP_SYS_BOOT) || kexec_load_disabled)
        return -EPERM;

    /*
     * Verify we have a legal set of flags
     * This leaves us room for future extensions.
     */
    if ((flags & KEXEC_FLAGS) != (flags & ~KEXEC_ARCH_MASK))
        return -EINVAL;

    /* Verify we are on the appropriate architecture */
    if (((flags & KEXEC_ARCH_MASK) != KEXEC_ARCH) &&
            ((flags & KEXEC_ARCH_MASK) != KEXEC_ARCH_DEFAULT))
        return -EINVAL;

    /* Put an artificial cap on the number
     * of segments passed to kexec_load.
     */
    if (nr_segments > KEXEC_SEGMENT_MAX)
        return -EINVAL;

    image = NULL;
    result = 0;

    /* Because we write directly to the reserved memory
     * region when loading crash kernels we need a mutex here to
     * prevent multiple crash  kernels from attempting to load
     * simultaneously, and to prevent a crash kernel from loading
     * over the top of a in use crash kernel.
     *
     * KISS: always take the mutex.
     */
    if (!mutex_trylock(&kexec_mutex))
        return -EBUSY;

    dest_image = &kexec_image;
    if (flags & KEXEC_ON_CRASH)
        dest_image = &kexec_crash_image;
    if (nr_segments > 0) {
        unsigned long i;

        if (flags & KEXEC_ON_CRASH) {
            /*
             * Loading another kernel to switch to if this one
             * crashes.  Free any current crash dump kernel before
             * we corrupt it.
             */

            kimage_free(xchg(&kexec_crash_image, NULL));
            result = kimage_alloc_init(&image, entry, nr_segments,
                                       segments, flags);
            crash_map_reserved_pages();
        } else {
            /* Loading another kernel to reboot into. */

            result = kimage_alloc_init(&image, entry, nr_segments,
                                       segments, flags);
        }
        if (result)
            goto out;

        if (flags & KEXEC_PRESERVE_CONTEXT)
            image->preserve_context = 1;
        result = machine_kexec_prepare(image);
        if (result)
            goto out;

        for (i = 0; i < nr_segments; i++) {
            result = kimage_load_segment(image, &image->segment[i]);
            if (result)
                goto out;
        }
        kimage_terminate(image);
        if (flags & KEXEC_ON_CRASH)
            crash_unmap_reserved_pages();
    }
    /* Install the new kernel, and  Uninstall the old */
    image = xchg(dest_image, image);

out:
    mutex_unlock(&kexec_mutex);
    kimage_free(image);

    return result;
}