cregit-Linux how code gets into the kernel

Release 4.11 arch/x86/kernel/machine_kexec_32.c

Directory: arch/x86/kernel
/*
 * handle transition of Linux booting another kernel
 * Copyright (C) 2002-2005 Eric Biederman  <ebiederm@xmission.com>
 *
 * This source code is licensed under the GNU General Public License,
 * Version 2.  See the file COPYING for more details.
 */

#include <linux/mm.h>
#include <linux/kexec.h>
#include <linux/delay.h>
#include <linux/numa.h>
#include <linux/ftrace.h>
#include <linux/suspend.h>
#include <linux/gfp.h>
#include <linux/io.h>

#include <asm/pgtable.h>
#include <asm/pgalloc.h>
#include <asm/tlbflush.h>
#include <asm/mmu_context.h>
#include <asm/apic.h>
#include <asm/io_apic.h>
#include <asm/cpufeature.h>
#include <asm/desc.h>
#include <asm/cacheflush.h>
#include <asm/debugreg.h>


static void set_idt(void *newidt, __u16 limit) { struct desc_ptr curidt; /* ia32 supports unaliged loads & stores */ curidt.size = limit; curidt.address = (unsigned long)newidt; load_idt(&curidt); }

Contributors

PersonTokensPropCommitsCommitProp
Eric W. Biedermann3382.50%250.00%
Zachary Amsden615.00%125.00%
Glauber de Oliveira Costa12.50%125.00%
Total40100.00%4100.00%


static void set_gdt(void *newgdt, __u16 limit) { struct desc_ptr curgdt; /* ia32 supports unaligned loads & stores */ curgdt.size = limit; curgdt.address = (unsigned long)newgdt; load_gdt(&curgdt); }

Contributors

PersonTokensPropCommitsCommitProp
Eric W. Biedermann3382.50%250.00%
Zachary Amsden615.00%125.00%
Glauber de Oliveira Costa12.50%125.00%
Total40100.00%4100.00%


static void load_segments(void) { #define __STR(X) #X #define STR(X) __STR(X) __asm__ __volatile__ ( "\tljmp $"STR(__KERNEL_CS)",$1f\n" "\t1:\n" "\tmovl $"STR(__KERNEL_DS)",%%eax\n" "\tmovl %%eax,%%ds\n" "\tmovl %%eax,%%es\n" "\tmovl %%eax,%%fs\n" "\tmovl %%eax,%%gs\n" "\tmovl %%eax,%%ss\n" : : : "eax", "memory"); #undef STR #undef __STR }

Contributors

PersonTokensPropCommitsCommitProp
Eric W. Biedermann3096.77%150.00%
Huang Ying13.23%150.00%
Total31100.00%2100.00%


static void machine_kexec_free_page_tables(struct kimage *image) { free_page((unsigned long)image->arch.pgd); #ifdef CONFIG_X86_PAE free_page((unsigned long)image->arch.pmd0); free_page((unsigned long)image->arch.pmd1); #endif free_page((unsigned long)image->arch.pte0); free_page((unsigned long)image->arch.pte1); }

Contributors

PersonTokensPropCommitsCommitProp
Huang Ying81100.00%1100.00%
Total81100.00%1100.00%


static int machine_kexec_alloc_page_tables(struct kimage *image) { image->arch.pgd = (pgd_t *)get_zeroed_page(GFP_KERNEL); #ifdef CONFIG_X86_PAE image->arch.pmd0 = (pmd_t *)get_zeroed_page(GFP_KERNEL); image->arch.pmd1 = (pmd_t *)get_zeroed_page(GFP_KERNEL); #endif image->arch.pte0 = (pte_t *)get_zeroed_page(GFP_KERNEL); image->arch.pte1 = (pte_t *)get_zeroed_page(GFP_KERNEL); if (!image->arch.pgd || #ifdef CONFIG_X86_PAE !image->arch.pmd0 || !image->arch.pmd1 || #endif !image->arch.pte0 || !image->arch.pte1) { machine_kexec_free_page_tables(image); return -ENOMEM; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Huang Ying147100.00%1100.00%
Total147100.00%1100.00%


static void machine_kexec_page_table_set_one( pgd_t *pgd, pmd_t *pmd, pte_t *pte, unsigned long vaddr, unsigned long paddr) { pud_t *pud; pgd += pgd_index(vaddr); #ifdef CONFIG_X86_PAE if (!(pgd_val(*pgd) & _PAGE_PRESENT)) set_pgd(pgd, __pgd(__pa(pmd) | _PAGE_PRESENT)); #endif pud = pud_offset(pgd, vaddr); pmd = pmd_offset(pud, vaddr); if (!(pmd_val(*pmd) & _PAGE_PRESENT)) set_pmd(pmd, __pmd(__pa(pte) | _PAGE_TABLE)); pte = pte_offset_kernel(pmd, vaddr); set_pte(pte, pfn_pte(paddr >> PAGE_SHIFT, PAGE_KERNEL_EXEC)); }

Contributors

PersonTokensPropCommitsCommitProp
Huang Ying139100.00%1100.00%
Total139100.00%1100.00%


static void machine_kexec_prepare_page_tables(struct kimage *image) { void *control_page; pmd_t *pmd = NULL; control_page = page_address(image->control_code_page); #ifdef CONFIG_X86_PAE pmd = image->arch.pmd0; #endif machine_kexec_page_table_set_one( image->arch.pgd, pmd, image->arch.pte0, (unsigned long)control_page, __pa(control_page)); #ifdef CONFIG_X86_PAE pmd = image->arch.pmd1; #endif machine_kexec_page_table_set_one( image->arch.pgd, pmd, image->arch.pte1, __pa(control_page), __pa(control_page)); }

Contributors

PersonTokensPropCommitsCommitProp
Huang Ying11099.10%150.00%
Hannes Eder10.90%150.00%
Total111100.00%2100.00%

/* * A architecture hook called to validate the * proposed image and prepare the control pages * as needed. The pages for KEXEC_CONTROL_PAGE_SIZE * have been allocated, but the segments have yet * been copied into the kernel. * * Do what every setup is needed on image and the * reboot code buffer to allow us to avoid allocations * later. * * - Make control page executable. * - Allocate page tables * - Setup page tables */
int machine_kexec_prepare(struct kimage *image) { int error; set_pages_x(image->control_code_page, 1); error = machine_kexec_alloc_page_tables(image); if (error) return error; machine_kexec_prepare_page_tables(image); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Huang Ying3375.00%375.00%
Eric W. Biedermann1125.00%125.00%
Total44100.00%4100.00%

/* * Undo anything leftover by machine_kexec_prepare * when an image is freed. */
void machine_kexec_cleanup(struct kimage *image) { set_pages_nx(image->control_code_page, 1); machine_kexec_free_page_tables(image); }

Contributors

PersonTokensPropCommitsCommitProp
Huang Ying1666.67%266.67%
Eric W. Biedermann833.33%133.33%
Total24100.00%3100.00%

/* * Do not allocate memory (or fail in any way) in machine_kexec(). * We are past the point of no return, committed to rebooting now. */
void machine_kexec(struct kimage *image) { unsigned long page_list[PAGES_NR]; void *control_page; int save_ftrace_enabled; asmlinkage unsigned long (*relocate_kernel_ptr)(unsigned long indirection_page, unsigned long control_page, unsigned long start_address, unsigned int has_pae, unsigned int preserve_context); #ifdef CONFIG_KEXEC_JUMP if (image->preserve_context) save_processor_state(); #endif save_ftrace_enabled = __ftrace_enabled_save(); /* Interrupts aren't acceptable while we reboot */ local_irq_disable(); hw_breakpoint_disable(); if (image->preserve_context) { #ifdef CONFIG_X86_IO_APIC /* * We need to put APICs in legacy mode so that we can * get timer interrupts in second kernel. kexec/kdump * paths already have calls to disable_IO_APIC() in * one form or other. kexec jump path also need * one. */ disable_IO_APIC(); #endif } control_page = page_address(image->control_code_page); memcpy(control_page, relocate_kernel, KEXEC_CONTROL_CODE_MAX_SIZE); relocate_kernel_ptr = control_page; page_list[PA_CONTROL_PAGE] = __pa(control_page); page_list[VA_CONTROL_PAGE] = (unsigned long)control_page; page_list[PA_PGD] = __pa(image->arch.pgd); if (image->type == KEXEC_TYPE_DEFAULT) page_list[PA_SWAP_PAGE] = (page_to_pfn(image->swap_page) << PAGE_SHIFT); /* * The segment registers are funny things, they have both a * visible and an invisible part. Whenever the visible part is * set to a specific selector, the invisible part is loaded * with from a table in memory. At no other time is the * descriptor table in memory accessed. * * I take advantage of this here by force loading the * segments, before I zap the gdt with an invalid value. */ load_segments(); /* * The gdt & idt are now invalid. * If you want to load them you must set up your own idt & gdt. */ set_gdt(phys_to_virt(0), 0); set_idt(phys_to_virt(0), 0); /* now call it */ image->start = relocate_kernel_ptr((unsigned long)image->head, (unsigned long)page_list, image->start, boot_cpu_has(X86_FEATURE_PAE), image->preserve_context); #ifdef CONFIG_KEXEC_JUMP if (image->preserve_context) restore_processor_state(); #endif __ftrace_enabled_restore(save_ftrace_enabled); }

Contributors

PersonTokensPropCommitsCommitProp
Huang Ying12348.05%753.85%
Eric W. Biedermann6625.78%17.69%
Magnus Damm5019.53%17.69%
Ken'ichi Ohmichi83.12%17.69%
Dave Hansen41.56%17.69%
K.Prasad31.17%17.69%
Ingo Molnar20.78%17.69%
Total256100.00%13100.00%


void arch_crash_save_vmcoreinfo(void) { #ifdef CONFIG_NUMA VMCOREINFO_SYMBOL(node_data); VMCOREINFO_LENGTH(node_data, MAX_NUMNODES); #endif #ifdef CONFIG_X86_PAE VMCOREINFO_CONFIG(X86_PAE); #endif }

Contributors

PersonTokensPropCommitsCommitProp
Ken'ichi Ohmichi34100.00%3100.00%
Total34100.00%3100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
Huang Ying66365.97%833.33%
Eric W. Biedermann21121.00%28.33%
Magnus Damm504.98%14.17%
Ken'ichi Ohmichi454.48%416.67%
Zachary Amsden141.39%28.33%
K.Prasad60.60%14.17%
Ingo Molnar50.50%14.17%
Dave Hansen40.40%14.17%
Jiang Liu30.30%14.17%
Glauber de Oliveira Costa20.20%14.17%
Dave Jones10.10%14.17%
Hannes Eder10.10%14.17%
Total1005100.00%24100.00%
Directory: arch/x86/kernel
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.