-
-
Notifications
You must be signed in to change notification settings - Fork 1.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
How to access the fan? #1350
Comments
The firmware controls the fan speed, not the OS. What the OS can do however is request the frequency of the CPU to change depending on how much cpu usage there is. For this it needs to follow the ACPI specification to request the firmware/cpu to change the frequency. |
Laptops may use Lithium battery, which is unstable; So the user should control the power quantity stored in Lithium battery, some researchers say that power around 50% is good for Lithium battery. But Sodium-ion battery is more stable. Can the OS control the power stored in Lithium battery, .e.g, between 45% and 55%? Sodium-based batteries are more stable so it's not important to control power stored in Sodium-based batteries. |
Depends on the laptop. There is an embedded controller on the mainboard which controls the charging logic. On some laptops there is an option to set a charging limit through ACPI, but on others there is no way to set a limit at all. |
How to find the memory address (if the ACPI allows) to set charging limit? |
There is no specific memory address. Instead you have to interpret AML bytecode that is part of the ACPI tables to set the charging limit. This AML bytecode may write to a memory address or it may trigger an interrupt calling into the firmware depending on how the firmware implemented setting the charging limit. The crates in https://github.com/rust-osdev/acpi may be useful for interacting with ACPI. Just be aware that the specification is a mess and basically everyone has to emulate all bugs in the Windows ACPI implementation to be compatible with common firmware implementations. https://github.com/rust-osdev/acpi does not yet do this (even Linux has trouble with getting everything exactly right), so it may not work on your specific system. You may also want to look at https://github.com/acpica/acpica which is the code Linux uses to interface with ACPI. There are no Rust bindings for this yet though afaik. |
If the AML contains newlines, should I write \n escape character or directly press Enter key? |
AML is a binary format. It is not a string, so it doesn't contain newlines. |
Brave search engine provides AI to answer questions, I searched
The AI said:
Is it correct? |
That is the text representation of AML. The actual ACPI tables contain AML in a binary format which is compiled from this text representation. |
Can the parse_table function https://docs.rs/aml/latest/aml/struct.AmlContext.html#method.parse_table parse the text representation to set charging limit? |
No, the aml crate doesn't have a compiler for the text representation of AML afaik. You shouldn't need to use the text representation of AML at any point. Instead read the binary version directly from the ACPI tables provided by the firmware of your system. Each firmware will provide different AML specific for the machine it runs on. |
Is there a universal way to set battery limit? How to know the length to read ACPI? It seems bootloader_api::info::BootInfo only provides rsdp_addr meaning the address. |
rsdp_addr points to the RSDP1, which itself contains the address of the RSDT (ACPI v1) and XSDT (ACPI v2)[^2]. The RSDT/XSDT is the index you can use to find all other ACPI tables. If you use the acpi crate you can use acpi::AcpiTables::from_rsdp to iterate over all ACPI tables. Footnotes |
As @bjorn3 said, it depends on the laptop. Check if the OS that the laptop was designed for (such as Windows or Linux) has a way of controlling battery charging. I know for some ThinkPads you can set a charging limit so it only charges up to a certain %. On most Chromebooks you can stop charging, set a charging current limit, and for some Chromebooks you can set a % limit. I think you would have to communicate with the laptop's embedded controller and send commands to it to control charging (and on Chromebooks you can control the fan too). I think this is pretty complicated to do in your own OS and you would have to look at the Linux kernel's code. |
Not to be a party pooper, but you have to be careful when learning or borrowing from the linux kernel. It's GPL licensed, so your kernel would need a compatible license (or you do your own stuff). Also mentioned here https://wiki.osdev.org/Linux_Kernel_Primer |
from_rsdp needs a handler, which needs PhysicalMapping. |
You are supposed to implement the |
Is BootInfo::kernel_len the length of my kernel? Can I put user data immediately at kernel_addr + kernel_len? |
Yes
No, that location may be reserved by the firmware or hardware. You have to iterate through the memory map in |
Why are start and end in MemoryRegion u64 but not usize? |
Not sure of that is the reason, but the page tables use 64bit entries when PAE is enabled, not pointer size entries. The distinction doesn't matter on x86_64 where PAE is always enabled and usize is always 64bit, but it avoids some casting between u64 and usize. On 32bit x86 the distinction does matter however. |
Does it ensure all entries in MemoryRegions are different so we don't need to worry about if there are 2 Usable entries? |
I'm not sure I fully understand the question, so I'll do my best. There's going to be more than 2 usable entries. Output in my osTRACE [kernel::mem::physical::physical_stage2] memory region: 0x0 - 0xa0000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x100000 - 0x156000 (Bootloader) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x156000 - 0x800000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x800000 - 0x808000 (UnknownUefi(10)) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x808000 - 0x80b000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x80b000 - 0x80c000 (UnknownUefi(10)) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x80c000 - 0x810000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x810000 - 0x900000 (UnknownUefi(10)) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x900000 - 0x1500000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x1500000 - 0x3f36000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x3f36000 - 0x3f56000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x3f56000 - 0x4e1d000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x4e1d000 - 0x6246000 (Bootloader) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6246000 - 0x6273000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6273000 - 0x629b000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x629b000 - 0x629d000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x629d000 - 0x62a5000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x62a5000 - 0x62a6000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x62a6000 - 0x62ac000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x62ac000 - 0x62ad000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x62ad000 - 0x6a30000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6a30000 - 0x6ad7000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6ad7000 - 0x6b4a000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6b4a000 - 0x6b84000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6b84000 - 0x6b88000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6b88000 - 0x6c24000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6c24000 - 0x6c2b000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6c2b000 - 0x6c35000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6c35000 - 0x6c40000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6c40000 - 0x6c4c000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6c4c000 - 0x6c4f000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6c4f000 - 0x6c6c000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6c6c000 - 0x6c6d000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6c6d000 - 0x6cf5000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6cf5000 - 0x6d01000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d01000 - 0x6d0b000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d0b000 - 0x6d0d000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d0d000 - 0x6d2d000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d2d000 - 0x6d30000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d30000 - 0x6d42000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d42000 - 0x6d43000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d43000 - 0x6d48000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d48000 - 0x6d4a000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d4a000 - 0x6d4f000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d4f000 - 0x6d54000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d54000 - 0x6d80000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d80000 - 0x6d82000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d82000 - 0x6d8f000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d8f000 - 0x6d90000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6d90000 - 0x6dc0000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6dc0000 - 0x6dc3000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6dc3000 - 0x6de1000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6de1000 - 0x6de9000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6de9000 - 0x6e00000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x6e00000 - 0x7005000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7005000 - 0x7013000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7013000 - 0x7015000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7015000 - 0x7037000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7037000 - 0x703b000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x703b000 - 0x7048000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7048000 - 0x7050000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7050000 - 0x705a000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x705a000 - 0x705d000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x705d000 - 0x706f000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x706f000 - 0x7076000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7076000 - 0x70a7000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x70a7000 - 0x70ab000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x70ab000 - 0x70ae000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x70ae000 - 0x70af000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x70af000 - 0x70b8000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x70b8000 - 0x70bb000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x70bb000 - 0x70d4000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x70d4000 - 0x70e1000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x70e1000 - 0x70e7000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x70e7000 - 0x74e7000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x74e7000 - 0x74ee000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x74ee000 - 0x74f1000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x74f1000 - 0x74f9000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x74f9000 - 0x74fd000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x74fd000 - 0x7515000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7515000 - 0x78ef000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x78ef000 - 0x79ef000 (UnknownUefi(6)) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x79ef000 - 0x7aef000 (UnknownUefi(5)) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7aef000 - 0x7b6f000 (UnknownUefi(0)) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7b6f000 - 0x7b7f000 (UnknownUefi(9)) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7b7f000 - 0x7bff000 (UnknownUefi(10)) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7bff000 - 0x7e00000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7e00000 - 0x7edc000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7edc000 - 0x7efc000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7efc000 - 0x7f24000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7f24000 - 0x7f2d000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7f2d000 - 0x7f58000 (Usable) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7f58000 - 0x7f78000 (UnknownUefi(6)) TRACE [kernel::mem::physical::physical_stage2] memory region: 0x7f78000 - 0x8000000 (UnknownUefi(10)) None of those regions overlap. You can use any of the Also, make sure to understand and handle the difference between physical and virtual memory. The addresses in the memory regions are physical memory. There are different ways to map those.
An identity mapping is the most limiting but simplest, and it's enough to get you started. Even if it doesn't sound like it, what you're trying to do (accessing fans or other auxiliary devices) is really advanced. I'd suggest starting with a solid memory implementation (recursive mapping, address spaces, physical and virtual memory managers) and then going from there. It's going to take some time, but there are many great resources out there. |
Some computers have a fan to prevent the temperature from being too hot, how to access and control the fan speed?
The text was updated successfully, but these errors were encountered: