⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 longhaul.c

📁 linux 内核源代码
💻 C
📖 第 1 页 / 共 2 页
字号:
/* *  (C) 2001-2004  Dave Jones. <davej@codemonkey.org.uk> *  (C) 2002  Padraig Brady. <padraig@antefacto.com> * *  Licensed under the terms of the GNU GPL License version 2. *  Based upon datasheets & sample CPUs kindly provided by VIA. * *  VIA have currently 3 different versions of Longhaul. *  Version 1 (Longhaul) uses the BCR2 MSR at 0x1147. *   It is present only in Samuel 1 (C5A), Samuel 2 (C5B) stepping 0. *  Version 2 of longhaul is backward compatible with v1, but adds *   LONGHAUL MSR for purpose of both frequency and voltage scaling. *   Present in Samuel 2 (steppings 1-7 only) (C5B), and Ezra (C5C). *  Version 3 of longhaul got renamed to Powersaver and redesigned *   to use only the POWERSAVER MSR at 0x110a. *   It is present in Ezra-T (C5M), Nehemiah (C5X) and above. *   It's pretty much the same feature wise to longhaul v2, though *   there is provision for scaling FSB too, but this doesn't work *   too well in practice so we don't even try to use this. * *  BIG FAT DISCLAIMER: Work in progress code. Possibly *dangerous* */#include <linux/kernel.h>#include <linux/module.h>#include <linux/moduleparam.h>#include <linux/init.h>#include <linux/cpufreq.h>#include <linux/pci.h>#include <linux/slab.h>#include <linux/string.h>#include <linux/delay.h>#include <asm/msr.h>#include <asm/timex.h>#include <asm/io.h>#include <asm/acpi.h>#include <linux/acpi.h>#include <acpi/processor.h>#include "longhaul.h"#define PFX "longhaul: "#define TYPE_LONGHAUL_V1	1#define TYPE_LONGHAUL_V2	2#define TYPE_POWERSAVER		3#define	CPU_SAMUEL	1#define	CPU_SAMUEL2	2#define	CPU_EZRA	3#define	CPU_EZRA_T	4#define	CPU_NEHEMIAH	5#define	CPU_NEHEMIAH_C	6/* Flags */#define USE_ACPI_C3		(1 << 1)#define USE_NORTHBRIDGE		(1 << 2)static int cpu_model;static unsigned int numscales=16;static unsigned int fsb;static const struct mV_pos *vrm_mV_table;static const unsigned char *mV_vrm_table;static unsigned int highest_speed, lowest_speed; /* kHz */static unsigned int minmult, maxmult;static int can_scale_voltage;static struct acpi_processor *pr = NULL;static struct acpi_processor_cx *cx = NULL;static u32 acpi_regs_addr;static u8 longhaul_flags;static unsigned int longhaul_index;/* Module parameters */static int scale_voltage;static int disable_acpi_c3;static int revid_errata;#define dprintk(msg...) cpufreq_debug_printk(CPUFREQ_DEBUG_DRIVER, "longhaul", msg)/* Clock ratios multiplied by 10 */static int clock_ratio[32];static int eblcr_table[32];static int longhaul_version;static struct cpufreq_frequency_table *longhaul_table;#ifdef CONFIG_CPU_FREQ_DEBUGstatic char speedbuffer[8];static char *print_speed(int speed){	if (speed < 1000) {		snprintf(speedbuffer, sizeof(speedbuffer),"%dMHz", speed);		return speedbuffer;	}	if (speed%1000 == 0)		snprintf(speedbuffer, sizeof(speedbuffer),			"%dGHz", speed/1000);	else		snprintf(speedbuffer, sizeof(speedbuffer),			"%d.%dGHz", speed/1000, (speed%1000)/100);	return speedbuffer;}#endifstatic unsigned int calc_speed(int mult){	int khz;	khz = (mult/10)*fsb;	if (mult%10)		khz += fsb/2;	khz *= 1000;	return khz;}static int longhaul_get_cpu_mult(void){	unsigned long invalue=0,lo, hi;	rdmsr (MSR_IA32_EBL_CR_POWERON, lo, hi);	invalue = (lo & (1<<22|1<<23|1<<24|1<<25)) >>22;	if (longhaul_version==TYPE_LONGHAUL_V2 || longhaul_version==TYPE_POWERSAVER) {		if (lo & (1<<27))			invalue+=16;	}	return eblcr_table[invalue];}/* For processor with BCR2 MSR */static void do_longhaul1(unsigned int clock_ratio_index){	union msr_bcr2 bcr2;	rdmsrl(MSR_VIA_BCR2, bcr2.val);	/* Enable software clock multiplier */	bcr2.bits.ESOFTBF = 1;	bcr2.bits.CLOCKMUL = clock_ratio_index & 0xff;	/* Sync to timer tick */	safe_halt();	/* Change frequency on next halt or sleep */	wrmsrl(MSR_VIA_BCR2, bcr2.val);	/* Invoke transition */	ACPI_FLUSH_CPU_CACHE();	halt();	/* Disable software clock multiplier */	local_irq_disable();	rdmsrl(MSR_VIA_BCR2, bcr2.val);	bcr2.bits.ESOFTBF = 0;	wrmsrl(MSR_VIA_BCR2, bcr2.val);}/* For processor with Longhaul MSR */static void do_powersaver(int cx_address, unsigned int clock_ratio_index,			  unsigned int dir){	union msr_longhaul longhaul;	u32 t;	rdmsrl(MSR_VIA_LONGHAUL, longhaul.val);	/* Setup new frequency */	if (!revid_errata)		longhaul.bits.RevisionKey = longhaul.bits.RevisionID;	else		longhaul.bits.RevisionKey = 0;	longhaul.bits.SoftBusRatio = clock_ratio_index & 0xf;	longhaul.bits.SoftBusRatio4 = (clock_ratio_index & 0x10) >> 4;	/* Setup new voltage */	if (can_scale_voltage)		longhaul.bits.SoftVID = (clock_ratio_index >> 8) & 0x1f;	/* Sync to timer tick */	safe_halt();	/* Raise voltage if necessary */	if (can_scale_voltage && dir) {		longhaul.bits.EnableSoftVID = 1;		wrmsrl(MSR_VIA_LONGHAUL, longhaul.val);		/* Change voltage */		if (!cx_address) {			ACPI_FLUSH_CPU_CACHE();			halt();		} else {			ACPI_FLUSH_CPU_CACHE();			/* Invoke C3 */			inb(cx_address);			/* Dummy op - must do something useless after P_LVL3			 * read */			t = inl(acpi_gbl_FADT.xpm_timer_block.address);		}		longhaul.bits.EnableSoftVID = 0;		wrmsrl(MSR_VIA_LONGHAUL, longhaul.val);	}	/* Change frequency on next halt or sleep */	longhaul.bits.EnableSoftBusRatio = 1;	wrmsrl(MSR_VIA_LONGHAUL, longhaul.val);	if (!cx_address) {		ACPI_FLUSH_CPU_CACHE();		halt();	} else {		ACPI_FLUSH_CPU_CACHE();		/* Invoke C3 */		inb(cx_address);		/* Dummy op - must do something useless after P_LVL3 read */		t = inl(acpi_gbl_FADT.xpm_timer_block.address);	}	/* Disable bus ratio bit */	longhaul.bits.EnableSoftBusRatio = 0;	wrmsrl(MSR_VIA_LONGHAUL, longhaul.val);	/* Reduce voltage if necessary */	if (can_scale_voltage && !dir) {		longhaul.bits.EnableSoftVID = 1;		wrmsrl(MSR_VIA_LONGHAUL, longhaul.val);		/* Change voltage */		if (!cx_address) {			ACPI_FLUSH_CPU_CACHE();			halt();		} else {			ACPI_FLUSH_CPU_CACHE();			/* Invoke C3 */			inb(cx_address);			/* Dummy op - must do something useless after P_LVL3			 * read */			t = inl(acpi_gbl_FADT.xpm_timer_block.address);		}		longhaul.bits.EnableSoftVID = 0;		wrmsrl(MSR_VIA_LONGHAUL, longhaul.val);	}}/** * longhaul_set_cpu_frequency() * @clock_ratio_index : bitpattern of the new multiplier. * * Sets a new clock ratio. */static void longhaul_setstate(unsigned int table_index){	unsigned int clock_ratio_index;	int speed, mult;	struct cpufreq_freqs freqs;	unsigned long flags;	unsigned int pic1_mask, pic2_mask;	u16 bm_status = 0;	u32 bm_timeout = 1000;	unsigned int dir = 0;	clock_ratio_index = longhaul_table[table_index].index;	/* Safety precautions */	mult = clock_ratio[clock_ratio_index & 0x1f];	if (mult == -1)		return;	speed = calc_speed(mult);	if ((speed > highest_speed) || (speed < lowest_speed))		return;	/* Voltage transition before frequency transition? */	if (can_scale_voltage && longhaul_index < table_index)		dir = 1;	freqs.old = calc_speed(longhaul_get_cpu_mult());	freqs.new = speed;	freqs.cpu = 0; /* longhaul.c is UP only driver */	cpufreq_notify_transition(&freqs, CPUFREQ_PRECHANGE);	dprintk ("Setting to FSB:%dMHz Mult:%d.%dx (%s)\n",			fsb, mult/10, mult%10, print_speed(speed/1000));retry_loop:	preempt_disable();	local_irq_save(flags);	pic2_mask = inb(0xA1);	pic1_mask = inb(0x21);	/* works on C3. save mask. */	outb(0xFF,0xA1);	/* Overkill */	outb(0xFE,0x21);	/* TMR0 only */	/* Wait while PCI bus is busy. */	if (acpi_regs_addr && (longhaul_flags & USE_NORTHBRIDGE	    || ((pr != NULL) && pr->flags.bm_control))) {		bm_status = inw(acpi_regs_addr);		bm_status &= 1 << 4;		while (bm_status && bm_timeout) {			outw(1 << 4, acpi_regs_addr);			bm_timeout--;			bm_status = inw(acpi_regs_addr);			bm_status &= 1 << 4;		}	}	if (longhaul_flags & USE_NORTHBRIDGE) {		/* Disable AGP and PCI arbiters */		outb(3, 0x22);	} else if ((pr != NULL) && pr->flags.bm_control) {		/* Disable bus master arbitration */		acpi_set_register(ACPI_BITREG_ARB_DISABLE, 1);	}	switch (longhaul_version) {	/*	 * Longhaul v1. (Samuel[C5A] and Samuel2 stepping 0[C5B])	 * Software controlled multipliers only.	 */	case TYPE_LONGHAUL_V1:		do_longhaul1(clock_ratio_index);		break;	/*	 * Longhaul v2 appears in Samuel2 Steppings 1->7 [C5B] and Ezra [C5C]	 *	 * Longhaul v3 (aka Powersaver). (Ezra-T [C5M] & Nehemiah [C5N])	 * Nehemiah can do FSB scaling too, but this has never been proven	 * to work in practice.	 */	case TYPE_LONGHAUL_V2:	case TYPE_POWERSAVER:		if (longhaul_flags & USE_ACPI_C3) {			/* Don't allow wakeup */			acpi_set_register(ACPI_BITREG_BUS_MASTER_RLD, 0);			do_powersaver(cx->address, clock_ratio_index, dir);		} else {			do_powersaver(0, clock_ratio_index, dir);		}		break;	}	if (longhaul_flags & USE_NORTHBRIDGE) {		/* Enable arbiters */		outb(0, 0x22);	} else if ((pr != NULL) && pr->flags.bm_control) {		/* Enable bus master arbitration */		acpi_set_register(ACPI_BITREG_ARB_DISABLE, 0);	}	outb(pic2_mask,0xA1);	/* restore mask */	outb(pic1_mask,0x21);	local_irq_restore(flags);	preempt_enable();	freqs.new = calc_speed(longhaul_get_cpu_mult());	/* Check if requested frequency is set. */	if (unlikely(freqs.new != speed)) {		printk(KERN_INFO PFX "Failed to set requested frequency!\n");		/* Revision ID = 1 but processor is expecting revision key		 * equal to 0. Jumpers at the bottom of processor will change		 * multiplier and FSB, but will not change bits in Longhaul		 * MSR nor enable voltage scaling. */		if (!revid_errata) {			printk(KERN_INFO PFX "Enabling \"Ignore Revision ID\" "						"option.\n");			revid_errata = 1;			msleep(200);			goto retry_loop;		}		/* Why ACPI C3 sometimes doesn't work is a mystery for me.		 * But it does happen. Processor is entering ACPI C3 state,		 * but it doesn't change frequency. I tried poking various		 * bits in northbridge registers, but without success. */		if (longhaul_flags & USE_ACPI_C3) {			printk(KERN_INFO PFX "Disabling ACPI C3 support.\n");			longhaul_flags &= ~USE_ACPI_C3;			if (revid_errata) {				printk(KERN_INFO PFX "Disabling \"Ignore "						"Revision ID\" option.\n");				revid_errata = 0;			}			msleep(200);			goto retry_loop;		}		/* This shouldn't happen. Longhaul ver. 2 was reported not		 * working on processors without voltage scaling, but with		 * RevID = 1. RevID errata will make things right. Just		 * to be 100% sure. */		if (longhaul_version == TYPE_LONGHAUL_V2) {			printk(KERN_INFO PFX "Switching to Longhaul ver. 1\n");			longhaul_version = TYPE_LONGHAUL_V1;			msleep(200);			goto retry_loop;		}	}	/* Report true CPU frequency */	cpufreq_notify_transition(&freqs, CPUFREQ_POSTCHANGE);	if (!bm_timeout)		printk(KERN_INFO PFX "Warning: Timeout while waiting for idle PCI bus.\n");}/* * Centaur decided to make life a little more tricky. * Only longhaul v1 is allowed to read EBLCR BSEL[0:1]. * Samuel2 and above have to try and guess what the FSB is. * We do this by assuming we booted at maximum multiplier, and interpolate * between that value multiplied by possible FSBs and cpu_mhz which * was calculated at boot time. Really ugly, but no other way to do this. */#define ROUNDING	0xfstatic int guess_fsb(int mult){	int speed = cpu_khz / 1000;	int i;	int speeds[] = { 666, 1000, 1333, 2000 };	int f_max, f_min;	for (i = 0; i < 4; i++) {		f_max = ((speeds[i] * mult) + 50) / 100;		f_max += (ROUNDING / 2);		f_min = f_max - ROUNDING;		if ((speed <= f_max) && (speed >= f_min))			return speeds[i] / 10;	}	return 0;}static int __init longhaul_get_ranges(void){	unsigned int i, j, k = 0;	unsigned int ratio;	int mult;	/* Get current frequency */	mult = longhaul_get_cpu_mult();	if (mult == -1) {		printk(KERN_INFO PFX "Invalid (reserved) multiplier!\n");		return -EINVAL;	}	fsb = guess_fsb(mult);	if (fsb == 0) {		printk(KERN_INFO PFX "Invalid (reserved) FSB!\n");		return -EINVAL;	}	/* Get max multiplier - as we always did.	 * Longhaul MSR is usefull only when voltage scaling is enabled.	 * C3 is booting at max anyway. */	maxmult = mult;	/* Get min multiplier */	switch (cpu_model) {	case CPU_NEHEMIAH:		minmult = 50;		break;	case CPU_NEHEMIAH_C:		minmult = 40;		break;	default:		minmult = 30;		break;	}	dprintk ("MinMult:%d.%dx MaxMult:%d.%dx\n",		 minmult/10, minmult%10, maxmult/10, maxmult%10);	highest_speed = calc_speed(maxmult);	lowest_speed = calc_speed(minmult);	dprintk ("FSB:%dMHz  Lowest speed: %s   Highest speed:%s\n", fsb,		 print_speed(lowest_speed/1000),		 print_speed(highest_speed/1000));	if (lowest_speed == highest_speed) {		printk (KERN_INFO PFX "highestspeed == lowest, aborting.\n");		return -EINVAL;	}	if (lowest_speed > highest_speed) {		printk (KERN_INFO PFX "nonsense! lowest (%d > %d) !\n",			lowest_speed, highest_speed);		return -EINVAL;	}	longhaul_table = kmalloc((numscales + 1) * sizeof(struct cpufreq_frequency_table), GFP_KERNEL);	if(!longhaul_table)		return -ENOMEM;	for (j = 0; j < numscales; j++) {		ratio = clock_ratio[j];		if (ratio == -1)			continue;		if (ratio > maxmult || ratio < minmult)			continue;		longhaul_table[k].frequency = calc_speed(ratio);		longhaul_table[k].index	= j;		k++;	}	if (k <= 1) {		kfree(longhaul_table);		return -ENODEV;	}	/* Sort */	for (j = 0; j < k - 1; j++) {		unsigned int min_f, min_i;		min_f = longhaul_table[j].frequency;		min_i = j;		for (i = j + 1; i < k; i++) {			if (longhaul_table[i].frequency < min_f) {				min_f = longhaul_table[i].frequency;				min_i = i;			}		}		if (min_i != j) {			unsigned int temp;			temp = longhaul_table[j].frequency;			longhaul_table[j].frequency = longhaul_table[min_i].frequency;			longhaul_table[min_i].frequency = temp;			temp = longhaul_table[j].index;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -