Date: Sat, 05 Jan 2008 12:21:33 +0900 From: Takeharu KATO <takeharu1219@ybb.ne.jp> To: Takeharu KATO <takeharu1219@ybb.ne.jp> Cc: freebsd-acpi@FreeBSD.org Subject: Re: powerd doesn't decrease CPU frequency in some cases Message-ID: <477EF7BD.8040800@ybb.ne.jp> In-Reply-To: <477EF09E.60606@ybb.ne.jp> References: <476E8674.5000303@gmail.com> <477EF09E.60606@ybb.ne.jp>
next in thread | previous in thread | raw e-mail | index | archive | help
This is a multi-part message in MIME format. --------------050906000808070009080005 Content-Type: text/plain; charset=windows-1251; format=flowed Content-Transfer-Encoding: 7bit Hi, Sorry, I sent wrong patch, please apply this instead. Takeharu KATO wrote: > Hi, > > I met same problem on my Panasonic CF-R7 note book PC. > As far as I investigate, cpufreq(est driver) does not check setting > values for PERFCTL register which are reported from acpi_perf driver. > > According to comments in sys/i386/cpufreq/est.c, > the auther of the driver regards this as TODO thing > (see following lines.). > > -- sys/i386/cpufreq/est.c > for (i = 0; i < count; i++) { > /* > * TODO: Figure out validity checks for id16. Linux checks > * that the control and status values match. > */ > -- > > I wrote the patch to check id16 values as comments says. > As far as I test the patch, the patch fixes the problem. > > Would you try this patch? > > P.S. I've send-pr this problem with this patch in this > morning(kern/119350). > > Andrey wrote: >> Good time of the day. >> >> I've noticed that powerd isn't able to decrease CPU frequency on my >> laptop (HP Compaq 6710b) as soon as frequency gets highest level. >> >> I've pottered a bit in the sources and it seems found the root of the >> issue. >> So those who are interested in the subject let consider it. >> >> For instance my system reports the following frequency levels: >> >> [silent@beastie][/home/silent]sysctl dev.cpu.0.freq_levels >> dev.cpu.0.freq_levels: 2001/35000 2000/35000 1750/30625 1600/25000 >> 1400/21875 1200/16000 1050/14000 900/12000 800/14000 700/12250 >> 600/10500 500/8750 400/7000 300/5250 >> >> If I try to adjust current frequency to 2000 MHz then I'll get: >> [silent@beastie][/home/silent]sudo sysctl dev.cpu.0.freq=2000 >> dev.cpu.0.freq: 300 -> 2001 >> Let check: >> [silent@beastie][/home/silent]sysctl dev.cpu.0.freq >> dev.cpu.0.freq: 2001 >> >> Thus, as you can see, I have level "2000" which system reports me but >> actually I can't to adjust those one exactly because it silently >> becomes "2001" >> >> Well... If I'm not mistaken powerd calculates the current "CPU idle >> mark" and if it is more then adopted value (by default 90%) then it >> shifts CPU frequency value 1 step down. In my case powerd sticks at >> "2001". It is obvious because when powerd decreases CPU frequency from >> the highest frequency level we'll get the following scenario: >> >> +-----------------------+ >> | dev.cpu.0.freq=2001 +--<-+ >> +-----------+-----------+ | >> | | >> Y | >> +-----------+-----------+ | >> | "CPU idle" > 90% | | >> +-----------+-----------+ | >> | | >> Y ^ >> +-----------+-----------+ ^ >> | powerd shifts freq. | ^ >> | 1 step down: | | >> | "2001" -> "2000" | | >> +-----------+-----------+ | >> | | >> Y | >> +-----------+-----------+ | >> | actually we have here | | >> | dev.cpu.0.freq == 2001+-->-+ >> +-----------------------+ >> >> >> According to the things mentioned above I've came to conclusion that >> in my case it is not a good idea to rely on frequency levels reported >> by the system. (Also I saw many sysctl mibs (dev.cpu.0.freq) of many >> other people. And there were "strange" frequency levels like "2000" >> and "2001". Of course I can't state that their systems' behaviors fit >> my case. But still...) >> >> So the simple way out I see is to teach powerd recognize "fake" >> frequency levels. Here I suggest a very simple workaround (and may be >> quite ugly... sorry I'm not sure if it is my cup of tee) which allows >> me to overcome the issue. And I hope it can be useful for smb. else. >> >> Also I'd like to hear opinions of others. May be there exists another >> and simpler way to overcome an issue or even I've missed something or >> not aware of something. >> >> >> Thank you. >> >> >> -- >> Sincerely, >> Andrey Kosachenko >> >> >> ------------------------------------------------------------------------ >> >> _______________________________________________ >> freebsd-acpi@freebsd.org mailing list >> http://lists.freebsd.org/mailman/listinfo/freebsd-acpi >> To unsubscribe, send any mail to "freebsd-acpi-unsubscribe@freebsd.org" > > > ------------------------------------------------------------------------ > > _______________________________________________ > freebsd-acpi@freebsd.org mailing list > http://lists.freebsd.org/mailman/listinfo/freebsd-acpi > To unsubscribe, send any mail to "freebsd-acpi-unsubscribe@freebsd.org" --------------050906000808070009080005 Content-Type: text/plain; name="est-fix-v2.patch" Content-Transfer-Encoding: 7bit Content-Disposition: inline; filename="est-fix-v2.patch" --- sys.orig/i386/cpufreq/est.c 2006-05-11 17:35:44.000000000 +0000 +++ sys/i386/cpufreq/est.c 2008-01-05 12:08:41.359933175 +0000 @@ -902,6 +902,8 @@ static int est_set(device_t dev, const struct cf_setting *set); static int est_get(device_t dev, struct cf_setting *set); static int est_type(device_t dev, int *type); +static int est_set_id16(uint16_t id16, int need_check); +static void est_get_id16(uint16_t *id16_p); static device_method_t est_methods[] = { /* Device interface */ @@ -1068,7 +1070,6 @@ return (0); } - static int est_acpi_info(device_t dev, freq_info **freqs) { @@ -1076,7 +1077,8 @@ struct cf_setting *sets; freq_info *table; device_t perf_dev; - int count, error, i; + int count, error, i, idx; + uint16_t saved_id16; perf_dev = device_find_child(device_get_parent(dev), "acpi_perf", -1); if (perf_dev == NULL || !device_is_attached(perf_dev)) @@ -1098,19 +1100,38 @@ error = ENOMEM; goto out; } - for (i = 0; i < count; i++) { + for (i = 0, idx = 0; i < count; i++) { /* - * TODO: Figure out validity checks for id16. Linux checks - * that the control and status values match. + * Confirm id16 value is correct. */ - table[i].freq = sets[i].freq; - table[i].volts = sets[i].volts; - table[i].id16 = sets[i].spec[0]; - table[i].power = sets[i].power; + if (sets[i].freq > 0) { + + /* save current value */ + est_get_id16(&saved_id16); + + /* + * Try to set specified value + */ + error = est_set_id16(sets[i].spec[0], 1); + if (error != 0) { + if (bootverbose) + printf("Invalid freq %u, ignored.\n", + sets[i].freq); + } else { + table[idx].freq = sets[i].freq; + table[idx].volts = sets[i].volts; + table[idx].id16 = sets[i].spec[0]; + table[idx].power = sets[i].power; + ++idx; + } + + /* restore saved setting */ + est_set_id16(sets[i].spec[0], 0); + } } /* Mark end of table with a terminator. */ - bzero(&table[i], sizeof(freq_info)); + bzero(&table[idx], sizeof(freq_info)); sc->acpi_settings = TRUE; *freqs = table; @@ -1148,6 +1169,41 @@ *freqs = p->freqtab; return (0); } +static void +est_get_id16(uint16_t *id16_p) { + *id16_p = rdmsr(MSR_PERF_STATUS) & 0xffff; +} + +static int +est_set_id16(uint16_t id16, int need_check) { + uint64_t msr; + uint16_t new_id16; + int rc = 0; + + /* + * Try to set freq. + */ + + /* Read the current register, mask out the old, set the new id. */ + msr = rdmsr(MSR_PERF_CTL); + msr = (msr & ~0xffff) | id16; + wrmsr(MSR_PERF_CTL, msr); + + /* Wait a short while for the new setting. XXX Is this necessary? */ + DELAY(EST_TRANS_LAT); + + if (need_check) { + est_get_id16(&new_id16); + if (new_id16 != id16) { + if (bootverbose) + printf("Invalid id16 (set, cur) = (%u, %u)\n", + id16, new_id16); + rc = ENXIO; /* Can not set this value */ + } + } + + return (rc); +} static freq_info * est_get_current(freq_info *freq_list) @@ -1162,7 +1218,7 @@ * we get a temporary invalid result. */ for (i = 0; i < 5; i++) { - id16 = rdmsr(MSR_PERF_STATUS) & 0xffff; + est_get_id16(&id16); for (f = freq_list; f->id16 != 0; f++) { if (f->id16 == id16) return (f); @@ -1201,7 +1257,6 @@ { struct est_softc *sc; freq_info *f; - uint64_t msr; /* Find the setting matching the requested one. */ sc = device_get_softc(dev); @@ -1213,9 +1268,7 @@ return (EINVAL); /* Read the current register, mask out the old, set the new id. */ - msr = rdmsr(MSR_PERF_CTL); - msr = (msr & ~0xffff) | f->id16; - wrmsr(MSR_PERF_CTL, msr); + est_set_id16(f->id16, 0); /* Wait a short while for the new setting. XXX Is this necessary? */ DELAY(EST_TRANS_LAT); --------------050906000808070009080005--
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?477EF7BD.8040800>