blob: b20a7acfd5efe59e9825166510eb72e16166b9eb [file] [log] [blame]
// Copyright 2017 The Fuchsia Authors
//
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file or at
// https://opensource.org/licenses/MIT
#include "kernel/percpu.h"
#include <debug.h>
#include <lib/counters.h>
#include <lib/system-topology.h>
#include <arch/ops.h>
#include <fbl/alloc_checker.h>
#include <ffl/string.h>
#include <kernel/align.h>
#include <kernel/cpu_distance_map.h>
#include <kernel/lockdep.h>
#include <ktl/algorithm.h>
#include <ktl/unique_ptr.h>
#include <lk/init.h>
#include <lockdep/lockdep.h>
#include <ktl/enforce.h>
decltype(percpu::boot_processor_) percpu::boot_processor_{};
percpu* percpu::secondary_processors_{nullptr};
percpu* percpu::boot_index_[1]{percpu::boot_processor_.GetAddressUnchecked()};
percpu** percpu::processor_index_{percpu::boot_index_};
size_t percpu::processor_count_{1};
percpu::percpu(cpu_num_t cpu_num) {
scheduler.this_cpu_ = cpu_num;
#if WITH_LOCK_DEP
// Initialize the lockdep tracking state for irq context.
auto* state = reinterpret_cast<lockdep::ThreadLockState*>(&lock_state);
lockdep::SystemInitThreadLockState(state);
#endif
counters = CounterArena().CpuData(cpu_num);
}
void percpu::InitializeBoot() {
boot_processor_.Initialize(0);
// Install a pointer to the boot CPU's high-level percpu struct in its low-level percpu struct.
arch_setup_percpu(0, &boot_processor_.Get());
}
void percpu::InitializeSecondariesBegin(uint32_t /*init_level*/) {
processor_count_ = CpuDistanceMap::Get().cpu_count();
DEBUG_ASSERT(processor_count_ != 0);
const size_t index_size = sizeof(percpu*) * processor_count_;
processor_index_ = static_cast<percpu**>(memalign(MAX_CACHE_LINE, index_size));
processor_index_[0] = &boot_processor_.Get();
static_assert((MAX_CACHE_LINE % alignof(struct percpu)) == 0);
const size_t bytes = sizeof(percpu) * (processor_count_ - 1);
if (bytes) {
secondary_processors_ = static_cast<percpu*>(memalign(MAX_CACHE_LINE, bytes));
// TODO: Remove the need to zero memory by fully initializing all of percpu
// members in the constructor / default initializers.
memset(secondary_processors_, 0, bytes);
// Construct the secondary percpu instances and add them to the index.
for (cpu_num_t i = 1; i < processor_count_; i++) {
processor_index_[i] = &secondary_processors_[i - 1];
new (&secondary_processors_[i - 1]) percpu{i};
}
}
// Compute the performance scale of each CPU.
{
fbl::AllocChecker checker;
ktl::unique_ptr<uint8_t[]> performance_class{new (&checker) uint8_t[processor_count_]};
if (checker.check()) {
uint8_t max_performance_class = 0;
for (cpu_num_t i = 0; i < processor_count_; i++) {
performance_class[i] = system_topology::GetPerformanceClass(i);
max_performance_class = ktl::max(performance_class[i], max_performance_class);
}
dprintf(INFO, "CPU performance scales:\n");
for (cpu_num_t i = 0; i < processor_count_; i++) {
const SchedPerformanceScale scale =
ffl::FromRatio(performance_class[i] + 1, max_performance_class + 1);
processor_index_[i]->scheduler.InitializePerformanceScale(scale);
CpuSearchSet::SetPerfScale(i, scale.raw_value());
dprintf(INFO, "CPU %2u: %s\n", i, Format(scale).c_str());
}
} else {
dprintf(INFO, "Failed to allocate temp buffer, using default performance for all CPUs\n");
}
}
// Determine the clusters before initializing the CPU search sets.
CpuSearchSet::AutoCluster(processor_count_);
CpuSearchSet::DumpClusters();
// Initialize the search set for each CPU.
dprintf(INFO, "CPU search order:\n");
for (cpu_num_t i = 0; i < processor_count_; i++) {
processor_index_[i]->search_set.Initialize(i, processor_count_);
processor_index_[i]->search_set.Dump();
const size_t cluster = processor_index_[i]->search_set.cluster();
processor_index_[i]->scheduler.cluster_ = cluster;
}
}
void percpu::InitializeSecondaryFinish() {
const cpu_num_t cpu_num = arch_curr_cpu_num();
DEBUG_ASSERT(cpu_num < processor_count());
arch_setup_percpu(cpu_num, processor_index_[cpu_num]);
}
// Allocate secondary percpu instances before booting other processors, after
// vm and system topology are initialized.
LK_INIT_HOOK(percpu_heap_init, percpu::InitializeSecondariesBegin, LK_INIT_LEVEL_KERNEL)