238 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			238 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "kernel/kernel.h"
 | |
| #include "kernel/watchdog.h"
 | |
| #include "arch_proto.h"
 | |
| #include "glo.h"
 | |
| #include <minix/minlib.h>
 | |
| #include <minix/u64.h>
 | |
| 
 | |
| #include "apic.h"
 | |
| 
 | |
| #define CPUID_UNHALTED_CORE_CYCLES_AVAILABLE	0
 | |
| 
 | |
| /*
 | |
|  * Intel architecture performance counters watchdog
 | |
|  */
 | |
| 
 | |
| static struct arch_watchdog intel_arch_watchdog;
 | |
| static struct arch_watchdog amd_watchdog;
 | |
| 
 | |
| static void intel_arch_watchdog_init(const unsigned cpu)
 | |
| {
 | |
| 	u64_t cpuf;
 | |
| 	u32_t val;
 | |
| 
 | |
| 	ia32_msr_write(INTEL_MSR_PERFMON_CRT0, 0, 0);
 | |
| 
 | |
| 	/* Int, OS, USR, Core ccyles */
 | |
| 	val = 1 << 20 | 1 << 17 | 1 << 16 | 0x3c;
 | |
| 	ia32_msr_write(INTEL_MSR_PERFMON_SEL0, 0, val);
 | |
| 
 | |
| 	/*
 | |
| 	 * should give as a tick approx. every 0.5-1s, the perf counter has only
 | |
| 	 * lowest 31 bits writable :(
 | |
| 	 */
 | |
| 	cpuf = cpu_get_freq(cpu);
 | |
| 	while (ex64hi(cpuf) || ex64lo(cpuf) > 0x7fffffffU)
 | |
| 		cpuf /= 2;
 | |
| 	cpuf = make64(-ex64lo(cpuf), ex64hi(cpuf));
 | |
| 	watchdog->resetval = watchdog->watchdog_resetval = cpuf;
 | |
| 
 | |
| 	ia32_msr_write(INTEL_MSR_PERFMON_CRT0, 0, ex64lo(cpuf));
 | |
| 
 | |
| 	ia32_msr_write(INTEL_MSR_PERFMON_SEL0, 0,
 | |
| 			val | INTEL_MSR_PERFMON_SEL0_ENABLE);
 | |
| 
 | |
| 	/* unmask the performance counter interrupt */
 | |
| 	lapic_write(LAPIC_LVTPCR, APIC_ICR_DM_NMI);
 | |
| }
 | |
| 
 | |
| static void intel_arch_watchdog_reinit(const unsigned cpu)
 | |
| {
 | |
| 	lapic_write(LAPIC_LVTPCR, APIC_ICR_DM_NMI);
 | |
| 	ia32_msr_write(INTEL_MSR_PERFMON_CRT0, 0, ex64lo(watchdog->resetval));
 | |
| }
 | |
| 
 | |
| int arch_watchdog_init(void)
 | |
| {
 | |
| 	u32_t eax, ebx, ecx, edx;
 | |
| 	unsigned cpu = cpuid;
 | |
| 
 | |
| 	if (!lapic_addr) {
 | |
| 		printf("ERROR : Cannot use NMI watchdog if APIC is not enabled\n");
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (cpu_info[cpu].vendor == CPU_VENDOR_INTEL) {
 | |
| 		eax = 0xA;
 | |
| 
 | |
| 		_cpuid(&eax, &ebx, &ecx, &edx);
 | |
| 
 | |
| 		/* FIXME currently we support only watchdog based on the intel
 | |
| 		 * architectural performance counters. Some Intel CPUs don't have this
 | |
| 		 * feature
 | |
| 		 */
 | |
| 		if (ebx & (1 << CPUID_UNHALTED_CORE_CYCLES_AVAILABLE))
 | |
| 			return -1;
 | |
| 		if (!((((eax >> 8)) & 0xff) > 0))
 | |
| 			return -1;
 | |
| 
 | |
| 		watchdog = &intel_arch_watchdog;
 | |
| 	} else if (cpu_info[cpu].vendor == CPU_VENDOR_AMD) {
 | |
| 		if (cpu_info[cpu].family != 6 &&
 | |
| 				cpu_info[cpu].family != 15 &&
 | |
| 				cpu_info[cpu].family != 16 &&
 | |
| 				cpu_info[cpu].family != 17)
 | |
| 			return -1;
 | |
| 		else
 | |
| 			watchdog = &amd_watchdog;
 | |
| 	} else
 | |
| 		return -1;
 | |
| 
 | |
| 	/* Setup PC overflow as NMI for watchdog, it is masked for now */
 | |
| 	lapic_write(LAPIC_LVTPCR, APIC_ICR_INT_MASK | APIC_ICR_DM_NMI);
 | |
| 	(void) lapic_read(LAPIC_LVTPCR);
 | |
| 
 | |
| 	/* double check if LAPIC is enabled */
 | |
| 	if (lapic_addr && watchdog->init) {
 | |
| 		watchdog->init(cpuid);
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| void arch_watchdog_stop(void)
 | |
| {
 | |
| }
 | |
| 
 | |
| void arch_watchdog_lockup(const struct nmi_frame * frame)
 | |
| {
 | |
| 	printf("KERNEL LOCK UP\n"
 | |
| 			"eax    0x%08x\n"
 | |
| 			"ecx    0x%08x\n"
 | |
| 			"edx    0x%08x\n"
 | |
| 			"ebx    0x%08x\n"
 | |
| 			"ebp    0x%08x\n"
 | |
| 			"esi    0x%08x\n"
 | |
| 			"edi    0x%08x\n"
 | |
| 			"gs     0x%08x\n"
 | |
| 			"fs     0x%08x\n"
 | |
| 			"es     0x%08x\n"
 | |
| 			"ds     0x%08x\n"
 | |
| 			"pc     0x%08x\n"
 | |
| 			"cs     0x%08x\n"
 | |
| 			"eflags 0x%08x\n",
 | |
| 			frame->eax,
 | |
| 			frame->ecx,
 | |
| 			frame->edx,
 | |
| 			frame->ebx,
 | |
| 			frame->ebp,
 | |
| 			frame->esi,
 | |
| 			frame->edi,
 | |
| 			frame->gs,
 | |
| 			frame->fs,
 | |
| 			frame->es,
 | |
| 			frame->ds,
 | |
| 			frame->pc,
 | |
| 			frame->cs,
 | |
| 			frame->eflags
 | |
| 			);
 | |
| 	panic("Kernel lockup");
 | |
| }
 | |
| 
 | |
| int i386_watchdog_start(void)
 | |
| {
 | |
| 	if (arch_watchdog_init()) {
 | |
| 		printf("WARNING watchdog initialization "
 | |
| 				"failed! Disabled\n");
 | |
| 		watchdog_enabled = 0;
 | |
| 		return -1;
 | |
| 	}
 | |
| 	else
 | |
| 		BOOT_VERBOSE(printf("Watchdog enabled\n"););
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int intel_arch_watchdog_profile_init(const unsigned freq)
 | |
| {
 | |
| 	u64_t cpuf;
 | |
| 
 | |
| 	/* FIXME works only if all CPUs have the same freq */
 | |
| 	cpuf = cpu_get_freq(cpuid);
 | |
| 	cpuf /= freq;
 | |
| 
 | |
| 	/*
 | |
| 	 * if freq is too low and the cpu freq too high we may get in a range of
 | |
| 	 * insane value which cannot be handled by the 31bit CPU perf counter
 | |
| 	 */
 | |
| 	if (ex64hi(cpuf) != 0 || ex64lo(cpuf) > 0x7fffffffU) {
 | |
| 		printf("ERROR : nmi watchdog ticks exceed 31bits, use higher frequency\n");
 | |
| 		return EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	cpuf = make64(-ex64lo(cpuf), ex64hi(cpuf));
 | |
| 	watchdog->profile_resetval = cpuf;
 | |
| 
 | |
| 	return OK;
 | |
| }
 | |
| 
 | |
| static struct arch_watchdog intel_arch_watchdog = {
 | |
| 	/*.init = */		intel_arch_watchdog_init,
 | |
| 	/*.reinit = */		intel_arch_watchdog_reinit,
 | |
| 	/*.profile_init = */	intel_arch_watchdog_profile_init
 | |
| };
 | |
| 
 | |
| #define AMD_MSR_EVENT_SEL0		0xc0010000
 | |
| #define AMD_MSR_EVENT_CTR0		0xc0010004
 | |
| #define AMD_MSR_EVENT_SEL0_ENABLE	(1 << 22)
 | |
| 
 | |
| static void amd_watchdog_init(const unsigned cpu)
 | |
| {
 | |
| 	u64_t cpuf;
 | |
| 	u32_t val;
 | |
| 
 | |
| 	ia32_msr_write(AMD_MSR_EVENT_CTR0, 0, 0);
 | |
| 
 | |
| 	/* Int, OS, USR, Cycles cpu is running */
 | |
| 	val = 1 << 20 | 1 << 17 | 1 << 16 | 0x76;
 | |
| 	ia32_msr_write(AMD_MSR_EVENT_SEL0, 0, val);
 | |
| 
 | |
| 	cpuf = -cpu_get_freq(cpu);
 | |
| 	watchdog->resetval = watchdog->watchdog_resetval = cpuf;
 | |
| 
 | |
| 	ia32_msr_write(AMD_MSR_EVENT_CTR0,
 | |
| 		       ex64hi(watchdog->resetval), ex64lo(watchdog->resetval));
 | |
| 
 | |
| 	ia32_msr_write(AMD_MSR_EVENT_SEL0, 0,
 | |
| 			val | AMD_MSR_EVENT_SEL0_ENABLE);
 | |
| 
 | |
| 	/* unmask the performance counter interrupt */
 | |
| 	lapic_write(LAPIC_LVTPCR, APIC_ICR_DM_NMI);
 | |
| }
 | |
| 
 | |
| static void amd_watchdog_reinit(const unsigned cpu)
 | |
| {
 | |
| 	lapic_write(LAPIC_LVTPCR, APIC_ICR_DM_NMI);
 | |
| 	ia32_msr_write(AMD_MSR_EVENT_CTR0,
 | |
| 		       ex64hi(watchdog->resetval), ex64lo(watchdog->resetval));
 | |
| }
 | |
| 
 | |
| static int amd_watchdog_profile_init(const unsigned freq)
 | |
| {
 | |
| 	u64_t cpuf;
 | |
| 
 | |
| 	/* FIXME works only if all CPUs have the same freq */
 | |
| 	cpuf = cpu_get_freq(cpuid);
 | |
| 	cpuf = -cpuf / freq;
 | |
| 
 | |
| 	watchdog->profile_resetval = cpuf;
 | |
| 
 | |
| 	return OK;
 | |
| }
 | |
| 
 | |
| static struct arch_watchdog amd_watchdog = {
 | |
| 	/*.init = */		amd_watchdog_init,
 | |
| 	/*.reinit = */		amd_watchdog_reinit,
 | |
| 	/*.profile_init = */	amd_watchdog_profile_init
 | |
| };
 | 
