cregit-Linux how code gets into the kernel

Release 4.10 drivers/acpi/processor_core.c

Directory: drivers/acpi
/*
 * Copyright (C) 2005 Intel Corporation
 * Copyright (C) 2009 Hewlett-Packard Development Company, L.P.
 *
 *      Alex Chiang <achiang@hp.com>
 *      - Unified x86/ia64 implementations
 *
 * I/O APIC hotplug support
 *      Yinghai Lu <yinghai@kernel.org>
 *      Jiang Liu <jiang.liu@intel.com>
 */
#include <linux/export.h>
#include <linux/acpi.h>
#include <acpi/processor.h>


#define _COMPONENT		ACPI_PROCESSOR_COMPONENT
ACPI_MODULE_NAME("processor_core");


static struct acpi_table_madt *get_madt_table(void) { static struct acpi_table_madt *madt; static int read_madt; if (!read_madt) { if (ACPI_FAILURE(acpi_get_table(ACPI_SIG_MADT, 0, (struct acpi_table_header **)&madt))) madt = NULL; read_madt++; } return madt; }

Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu58100.00%1100.00%
Total58100.00%1100.00%


static int map_lapic_id(struct acpi_subtable_header *entry, u32 acpi_id, phys_cpuid_t *apic_id, bool ignore_disabled) { struct acpi_madt_local_apic *lapic = container_of(entry, struct acpi_madt_local_apic, header); if (ignore_disabled && !(lapic->lapic_flags & ACPI_MADT_ENABLED)) return -ENODEV; if (lapic->processor_id != acpi_id) return -EINVAL; *apic_id = lapic->id; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
alex chiangalex chiang5776.00%233.33%
fabian frederickfabian frederick79.33%116.67%
gu zhenggu zheng56.67%116.67%
hanjun guohanjun guo56.67%116.67%
catalin marinascatalin marinas11.33%116.67%
Total75100.00%6100.00%


static int map_x2apic_id(struct acpi_subtable_header *entry, int device_declaration, u32 acpi_id, phys_cpuid_t *apic_id, bool ignore_disabled) { struct acpi_madt_local_x2apic *apic = container_of(entry, struct acpi_madt_local_x2apic, header); if (ignore_disabled && !(apic->lapic_flags & ACPI_MADT_ENABLED)) return -ENODEV; if (device_declaration && (apic->uid == acpi_id)) { *apic_id = apic->local_apic_id; return 0; } return -EINVAL; }

Contributors

PersonTokensPropCommitsCommitProp
alex chiangalex chiang6678.57%233.33%
fabian frederickfabian frederick78.33%116.67%
gu zhenggu zheng55.95%116.67%
hanjun guohanjun guo55.95%116.67%
catalin marinascatalin marinas11.19%116.67%
Total84100.00%6100.00%


static int map_lsapic_id(struct acpi_subtable_header *entry, int device_declaration, u32 acpi_id, phys_cpuid_t *apic_id, bool ignore_disabled) { struct acpi_madt_local_sapic *lsapic = container_of(entry, struct acpi_madt_local_sapic, header); if (ignore_disabled && !(lsapic->lapic_flags & ACPI_MADT_ENABLED)) return -ENODEV; if (device_declaration) { if ((entry->length < 16) || (lsapic->uid != acpi_id)) return -EINVAL; } else if (lsapic->processor_id != acpi_id) return -EINVAL; *apic_id = (lsapic->id << 8) | lsapic->eid; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
alex chiangalex chiang9582.61%233.33%
fabian frederickfabian frederick76.09%116.67%
hanjun guohanjun guo76.09%116.67%
gu zhenggu zheng54.35%116.67%
catalin marinascatalin marinas10.87%116.67%
Total115100.00%6100.00%

/* * Retrieve the ARM CPU physical identifier (MPIDR) */
static int map_gicc_mpidr(struct acpi_subtable_header *entry, int device_declaration, u32 acpi_id, phys_cpuid_t *mpidr, bool ignore_disabled) { struct acpi_madt_generic_interrupt *gicc = container_of(entry, struct acpi_madt_generic_interrupt, header); if (ignore_disabled && !(gicc->flags & ACPI_MADT_ENABLED)) return -ENODEV; /* device_declaration means Device object in DSDT, in the * GIC interrupt model, logical processors are required to * have a Processor Device object in the DSDT, so we should * check device_declaration here */ if (device_declaration && (gicc->uid == acpi_id)) { *mpidr = gicc->arm_mpidr; return 0; } return -EINVAL; }

Contributors

PersonTokensPropCommitsCommitProp
hanjun guohanjun guo8094.12%150.00%
gu zhenggu zheng55.88%150.00%
Total85100.00%2100.00%


static phys_cpuid_t map_madt_entry(struct acpi_table_madt *madt, int type, u32 acpi_id, bool ignore_disabled) { unsigned long madt_end, entry; phys_cpuid_t phys_id = PHYS_CPUID_INVALID; /* CPU hardware ID */ if (!madt) return phys_id; entry = (unsigned long)madt; madt_end = entry + madt->header.length; /* Parse all entries looking for a match. */ entry += sizeof(struct acpi_table_madt); while (entry + sizeof(struct acpi_subtable_header) < madt_end) { struct acpi_subtable_header *header = (struct acpi_subtable_header *)entry; if (header->type == ACPI_MADT_TYPE_LOCAL_APIC) { if (!map_lapic_id(header, acpi_id, &phys_id, ignore_disabled)) break; } else if (header->type == ACPI_MADT_TYPE_LOCAL_X2APIC) { if (!map_x2apic_id(header, type, acpi_id, &phys_id, ignore_disabled)) break; } else if (header->type == ACPI_MADT_TYPE_LOCAL_SAPIC) { if (!map_lsapic_id(header, type, acpi_id, &phys_id, ignore_disabled)) break; } else if (header->type == ACPI_MADT_TYPE_GENERIC_INTERRUPT) { if (!map_gicc_mpidr(header, type, acpi_id, &phys_id, ignore_disabled)) break; } entry += header->length; } return phys_id; }

Contributors

PersonTokensPropCommitsCommitProp
alex chiangalex chiang15973.95%225.00%
hanjun guohanjun guo3717.21%337.50%
gu zhenggu zheng115.12%112.50%
david daneydavid daney52.33%112.50%
catalin marinascatalin marinas31.40%112.50%
Total215100.00%8100.00%


phys_cpuid_t __init acpi_map_madt_entry(u32 acpi_id) { struct acpi_table_madt *madt = NULL; phys_cpuid_t rv; acpi_get_table(ACPI_SIG_MADT, 0, (struct acpi_table_header **)&madt); if (!madt) return PHYS_CPUID_INVALID; rv = map_madt_entry(madt, 1, acpi_id, true); acpi_put_table((struct acpi_table_header *)madt); return rv; }

Contributors

PersonTokensPropCommitsCommitProp
david daneydavid daney6086.96%133.33%
lv zhenglv zheng710.14%133.33%
gu zhenggu zheng22.90%133.33%
Total69100.00%3100.00%


static phys_cpuid_t map_mat_entry(acpi_handle handle, int type, u32 acpi_id, bool ignore_disabled) { struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL }; union acpi_object *obj; struct acpi_subtable_header *header; phys_cpuid_t phys_id = PHYS_CPUID_INVALID; if (ACPI_FAILURE(acpi_evaluate_object(handle, "_MAT", NULL, &buffer))) goto exit; if (!buffer.length || !buffer.pointer) goto exit; obj = buffer.pointer; if (obj->type != ACPI_TYPE_BUFFER || obj->buffer.length < sizeof(struct acpi_subtable_header)) { goto exit; } header = (struct acpi_subtable_header *)obj->buffer.pointer; if (header->type == ACPI_MADT_TYPE_LOCAL_APIC) map_lapic_id(header, acpi_id, &phys_id, ignore_disabled); else if (header->type == ACPI_MADT_TYPE_LOCAL_SAPIC) map_lsapic_id(header, type, acpi_id, &phys_id, ignore_disabled); else if (header->type == ACPI_MADT_TYPE_LOCAL_X2APIC) map_x2apic_id(header, type, acpi_id, &phys_id, ignore_disabled); else if (header->type == ACPI_MADT_TYPE_GENERIC_INTERRUPT) map_gicc_mpidr(header, type, acpi_id, &phys_id, ignore_disabled); exit: kfree(buffer.pointer); return phys_id; }

Contributors

PersonTokensPropCommitsCommitProp
alex chiangalex chiang16373.09%116.67%
hanjun guohanjun guo4620.63%350.00%
gu zhenggu zheng114.93%116.67%
catalin marinascatalin marinas31.35%116.67%
Total223100.00%6100.00%


static phys_cpuid_t __acpi_get_phys_id(acpi_handle handle, int type, u32 acpi_id, bool ignore_disabled) { phys_cpuid_t phys_id; phys_id = map_mat_entry(handle, type, acpi_id, ignore_disabled); if (invalid_phys_cpuid(phys_id)) phys_id = map_madt_entry(get_madt_table(), type, acpi_id, ignore_disabled); return phys_id; }

Contributors

PersonTokensPropCommitsCommitProp
alex chiangalex chiang3356.90%114.29%
gu zhenggu zheng915.52%114.29%
hanjun guohanjun guo813.79%228.57%
david daneydavid daney35.17%114.29%
jiang liujiang liu35.17%114.29%
catalin marinascatalin marinas23.45%114.29%
Total58100.00%7100.00%


phys_cpuid_t acpi_get_phys_id(acpi_handle handle, int type, u32 acpi_id) { return __acpi_get_phys_id(handle, type, acpi_id, true); }

Contributors

PersonTokensPropCommitsCommitProp
gu zhenggu zheng26100.00%1100.00%
Total26100.00%1100.00%


int acpi_map_cpuid(phys_cpuid_t phys_id, u32 acpi_id) { #ifdef CONFIG_SMP int i; #endif if (invalid_phys_cpuid(phys_id)) { /* * On UP processor, there is no _MAT or MADT table. * So above phys_id is always set to PHYS_CPUID_INVALID. * * BIOS may define multiple CPU handles even for UP processor. * For example, * * Scope (_PR) * { * Processor (CPU0, 0x00, 0x00000410, 0x06) {} * Processor (CPU1, 0x01, 0x00000410, 0x06) {} * Processor (CPU2, 0x02, 0x00000410, 0x06) {} * Processor (CPU3, 0x03, 0x00000410, 0x06) {} * } * * Ignores phys_id and always returns 0 for the processor * handle with acpi id 0 if nr_cpu_ids is 1. * This should be the case if SMP tables are not found. * Return -EINVAL for other CPU's handle. */ if (nr_cpu_ids <= 1 && acpi_id == 0) return acpi_id; else return -EINVAL; } #ifdef CONFIG_SMP for_each_possible_cpu(i) { if (cpu_physical_id(i) == phys_id) return i; } #else /* In UP kernel, only processor 0 is valid */ if (phys_id == 0) return phys_id; #endif return -ENODEV; }

Contributors

PersonTokensPropCommitsCommitProp
lin minglin ming2731.40%222.22%
alex chiangalex chiang2630.23%111.11%
jiang liujiang liu1618.60%111.11%
hanjun guohanjun guo1112.79%333.33%
thomas renningerthomas renninger44.65%111.11%
catalin marinascatalin marinas22.33%111.11%
Total86100.00%9100.00%


int acpi_get_cpuid(acpi_handle handle, int type, u32 acpi_id) { phys_cpuid_t phys_id; phys_id = acpi_get_phys_id(handle, type, acpi_id); return acpi_map_cpuid(phys_id, acpi_id); }

Contributors

PersonTokensPropCommitsCommitProp
jiang liujiang liu3186.11%133.33%
hanjun guohanjun guo411.11%133.33%
catalin marinascatalin marinas12.78%133.33%
Total36100.00%3100.00%

EXPORT_SYMBOL_GPL(acpi_get_cpuid); #ifdef CONFIG_ACPI_HOTPLUG_CPU
static bool __init map_processor(acpi_handle handle, phys_cpuid_t *phys_id, int *cpuid) { int type, id; u32 acpi_id; acpi_status status; acpi_object_type acpi_type; unsigned long long tmp; union acpi_object object = { 0 }; struct acpi_buffer buffer = { sizeof(union acpi_object), &object }; status = acpi_get_type(handle, &acpi_type); if (ACPI_FAILURE(status)) return false; switch (acpi_type) { case ACPI_TYPE_PROCESSOR: status = acpi_evaluate_object(handle, NULL, NULL, &buffer); if (ACPI_FAILURE(status)) return false; acpi_id = object.processor.proc_id; /* validate the acpi_id */ if(acpi_processor_validate_proc_id(acpi_id)) return false; break; case ACPI_TYPE_DEVICE: status = acpi_evaluate_integer(handle, "_UID", NULL, &tmp); if (ACPI_FAILURE(status)) return false; acpi_id = tmp; break; default: return false; } type = (acpi_type == ACPI_TYPE_DEVICE) ? 1 : 0; *phys_id = __acpi_get_phys_id(handle, type, acpi_id, false); id = acpi_map_cpuid(*phys_id, acpi_id); if (id < 0) return false; *cpuid = id; return true; }

Contributors

PersonTokensPropCommitsCommitProp
gu zhenggu zheng20090.09%133.33%
thomas gleixnerthomas gleixner114.95%133.33%
dou liyangdou liyang114.95%133.33%
Total222100.00%3100.00%


static acpi_status __init set_processor_node_mapping(acpi_handle handle, u32 lvl, void *context, void **rv) { phys_cpuid_t phys_id; int cpu_id; if (!map_processor(handle, &phys_id, &cpu_id)) return AE_ERROR; acpi_map_cpu2node(handle, cpu_id, phys_id); return AE_OK; }

Contributors

PersonTokensPropCommitsCommitProp
gu zhenggu zheng57100.00%1100.00%
Total57100.00%1100.00%


void __init acpi_set_processor_mapping(void) { /* Set persistent cpu <-> node mapping for all processors. */ acpi_walk_namespace(ACPI_TYPE_PROCESSOR, ACPI_ROOT_OBJECT, ACPI_UINT32_MAX, set_processor_node_mapping, NULL, NULL, NULL); }

Contributors

PersonTokensPropCommitsCommitProp
gu zhenggu zheng26100.00%1100.00%
Total26100.00%1100.00%

#else
void __init acpi_set_processor_mapping(void) {}

Contributors

PersonTokensPropCommitsCommitProp
gu zhenggu zheng7100.00%1100.00%
Total7100.00%1100.00%

#endif /* CONFIG_ACPI_HOTPLUG_CPU */ #ifdef CONFIG_ACPI_HOTPLUG_IOAPIC
static int get_ioapic_id(struct acpi_subtable_header *entry, u32 gsi_base, u64 *phys_addr, int *ioapic_id) { struct acpi_madt_io_apic *ioapic = (struct acpi_madt_io_apic *)entry; if (ioapic->global_irq_base != gsi_base) return 0; *phys_addr = ioapic->address; *ioapic_id = ioapic->id; return 1; }

Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu62100.00%1100.00%
Total62100.00%1100.00%


static int parse_madt_ioapic_entry(u32 gsi_base, u64 *phys_addr) { struct acpi_subtable_header *hdr; unsigned long madt_end, entry; struct acpi_table_madt *madt; int apic_id = -1; madt = get_madt_table(); if (!madt) return apic_id; entry = (unsigned long)madt; madt_end = entry + madt->header.length; /* Parse all entries looking for a match. */ entry += sizeof(struct acpi_table_madt); while (entry + sizeof(struct acpi_subtable_header) < madt_end) { hdr = (struct acpi_subtable_header *)entry; if (hdr->type == ACPI_MADT_TYPE_IO_APIC && get_ioapic_id(hdr, gsi_base, phys_addr, &apic_id)) break; else entry += hdr->length; } return apic_id; }

Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu129100.00%1100.00%
Total129100.00%1100.00%


static int parse_mat_ioapic_entry(acpi_handle handle, u32 gsi_base, u64 *phys_addr) { struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL }; struct acpi_subtable_header *header; union acpi_object *obj; int apic_id = -1; if (ACPI_FAILURE(acpi_evaluate_object(handle, "_MAT", NULL, &buffer))) goto exit; if (!buffer.length || !buffer.pointer) goto exit; obj = buffer.pointer; if (obj->type != ACPI_TYPE_BUFFER || obj->buffer.length < sizeof(struct acpi_subtable_header)) goto exit; header = (struct acpi_subtable_header *)obj->buffer.pointer; if (header->type == ACPI_MADT_TYPE_IO_APIC) get_ioapic_id(header, gsi_base, phys_addr, &apic_id); exit: kfree(buffer.pointer); return apic_id; }

Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu151100.00%1100.00%
Total151100.00%1100.00%

/** * acpi_get_ioapic_id - Get IOAPIC ID and physical address matching @gsi_base * @handle: ACPI object for IOAPIC device * @gsi_base: GSI base to match with * @phys_addr: Pointer to store physical address of matching IOAPIC record * * Walk resources returned by ACPI_MAT method, then ACPI MADT table, to search * for an ACPI IOAPIC record matching @gsi_base. * Return IOAPIC id and store physical address in @phys_addr if found a match, * otherwise return <0. */
int acpi_get_ioapic_id(acpi_handle handle, u32 gsi_base, u64 *phys_addr) { int apic_id; apic_id = parse_mat_ioapic_entry(handle, gsi_base, phys_addr); if (apic_id == -1) apic_id = parse_madt_ioapic_entry(gsi_base, phys_addr); return apic_id; }

Contributors

PersonTokensPropCommitsCommitProp
yinghai luyinghai lu48100.00%1100.00%
Total48100.00%1100.00%

#endif /* CONFIG_ACPI_HOTPLUG_IOAPIC */

Overall Contributors

PersonTokensPropCommitsCommitProp
alex chiangalex chiang61833.01%725.00%
yinghai luyinghai lu45624.36%13.57%
gu zhenggu zheng37720.14%27.14%
hanjun guohanjun guo20410.90%621.43%
david daneydavid daney683.63%13.57%
jiang liujiang liu502.67%13.57%
lin minglin ming271.44%27.14%
fabian frederickfabian frederick211.12%13.57%
catalin marinascatalin marinas140.75%13.57%
dou liyangdou liyang110.59%13.57%
thomas gleixnerthomas gleixner110.59%13.57%
lv zhenglv zheng80.43%27.14%
thomas renningerthomas renninger40.21%13.57%
paul gortmakerpaul gortmaker30.16%13.57%
Total1872100.00%28100.00%
Directory: drivers/acpi
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.