mirror of
https://github.com/0xAX/linux-insides.git
synced 2025-01-03 12:20:56 +00:00
commit
c0e2896ac9
@ -89,10 +89,10 @@ endif
|
|||||||
|
|
||||||
Now we know where to start, so let's do it.
|
Now we know where to start, so let's do it.
|
||||||
|
|
||||||
Reload the segments if need
|
Reload the segments if needed
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
As i wrote above, we start in the [arch/x86/boot/compressed/head_64.S](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/head_64.S). First of all we can see before `startup_32` definition:
|
As I wrote above, we start in [arch/x86/boot/compressed/head_64.S](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/head_64.S). First of all we can see before the `startup_32` definition:
|
||||||
|
|
||||||
```assembly
|
```assembly
|
||||||
__HEAD
|
__HEAD
|
||||||
@ -100,7 +100,7 @@ As i wrote above, we start in the [arch/x86/boot/compressed/head_64.S](https://g
|
|||||||
ENTRY(startup_32)
|
ENTRY(startup_32)
|
||||||
```
|
```
|
||||||
|
|
||||||
`__HEAD` defined in the [include/linux/init.h](https://github.com/torvalds/linux/blob/master/include/linux/init.h) and looks as:
|
`__HEAD` is defined in [include/linux/init.h](https://github.com/torvalds/linux/blob/master/include/linux/init.h) and looks like:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
#define __HEAD .section ".head.text","ax"
|
#define __HEAD .section ".head.text","ax"
|
||||||
@ -119,17 +119,17 @@ SECTIONS
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
Note on `. = 0;`. `.` is a special variable of linker - location counter. Assigning a value to it, is an offset relative to the offset of the segment. As we assign zero to it, we can read from comments:
|
Note on `. = 0;`. `.` is a special variable of linker - location counter. The value assigned to it is an offset relative to the offset of the segment. As we assign zero to it, we can read from comments:
|
||||||
|
|
||||||
```
|
```
|
||||||
Be careful parts of head_64.S assume startup_32 is at address 0.
|
Be careful parts of head_64.S assume startup_32 is at address 0.
|
||||||
```
|
```
|
||||||
|
|
||||||
Ok, now we know where we are, and now the best time to look inside the `startup_32` function.
|
Ok, now we know where we are, and now is the best time to look inside the `startup_32` function.
|
||||||
|
|
||||||
In the start of the `startup_32` we can see the `cld` instruction which clears `DF` flag. After this, string operations like `stosb` and other will increment the index registers `esi` or `edi`.
|
In the start of `startup_32` we can see the `cld` instruction which clears the `DF` flag. After this, string operations like `stosb` and others will increment the index registers `esi` or `edi`.
|
||||||
|
|
||||||
The Next we can see the check of `KEEP_SEGMENTS` flag from `loadflags`. If you remember we already saw `loadflags` in the `arch/x86/boot/head.S` (there we checked flag `CAN_USE_HEAP`). Now we need to check `KEEP_SEGMENTS` flag. We can find description of this flag in the linux boot protocol:
|
Next we can see the check of the `KEEP_SEGMENTS` flag from `loadflags`. If you remember we already saw `loadflags` in the `arch/x86/boot/head.S` (there we checked flag `CAN_USE_HEAP`). Now we need to check the `KEEP_SEGMENTS` flag. We can find a description of this flag in the linux boot protocol:
|
||||||
|
|
||||||
```
|
```
|
||||||
Bit 6 (write): KEEP_SEGMENTS
|
Bit 6 (write): KEEP_SEGMENTS
|
||||||
@ -140,7 +140,7 @@ Bit 6 (write): KEEP_SEGMENTS
|
|||||||
a base of 0 (or the equivalent for their environment).
|
a base of 0 (or the equivalent for their environment).
|
||||||
```
|
```
|
||||||
|
|
||||||
and if `KEEP_SEGMENTS` is not set, we need to set `ds`, `ss` and `es` registers to flat segment with base 0. That we do:
|
and if `KEEP_SEGMENTS` is not set, we need to set `ds`, `ss` and `es` registers to a flat segment with base 0. That we do:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
testb $(1 << 6), BP_loadflags(%esi)
|
testb $(1 << 6), BP_loadflags(%esi)
|
||||||
@ -155,9 +155,9 @@ and if `KEEP_SEGMENTS` is not set, we need to set `ds`, `ss` and `es` registers
|
|||||||
|
|
||||||
remember that `__BOOT_DS` is `0x18` (index of data segment in the Global Descriptor Table). If `KEEP_SEGMENTS` is not set, we jump to the label `1f` or update segment registers with `__BOOT_DS` if this flag is set.
|
remember that `__BOOT_DS` is `0x18` (index of data segment in the Global Descriptor Table). If `KEEP_SEGMENTS` is not set, we jump to the label `1f` or update segment registers with `__BOOT_DS` if this flag is set.
|
||||||
|
|
||||||
If you read previous the [part](https://github.com/0xAX/linux-insides/blob/master/Booting/linux-bootstrap-3.md), you can remember that we already updated segment registers in the [arch/x86/boot/pmjump.S](https://github.com/torvalds/linux/blob/master/arch/x86/boot/pmjump.S), so why we need to set up it again? Actually linux kernel has also 32-bit boot protocol, so `startup_32` can be first function which will be executed right after a bootloader transfers control to the kernel.
|
If you read the previous [part](https://github.com/0xAX/linux-insides/blob/master/Booting/linux-bootstrap-3.md), you can remember that we already updated segment registers in the [arch/x86/boot/pmjump.S](https://github.com/torvalds/linux/blob/master/arch/x86/boot/pmjump.S), so why do we need to set up it again? Actually linux kernel also has the 32-bit boot protocol, so `startup_32` can be the first function which will be executed right after a bootloader transfers control to the kernel.
|
||||||
|
|
||||||
As we checked `KEEP_SEGMENTS` flag and put the correct value to the segment registers, next step is calculate difference between where we loaded and compiled to run (remember that `setup.ld.S` contains `. = 0` at the start of the section):
|
As we checked the `KEEP_SEGMENTS` flag and put the correct value to the segment registers, the next step is to calculate difference between where we loaded and compiled to run (remember that `setup.ld.S` contains `. = 0` at the start of the section):
|
||||||
|
|
||||||
```assembly
|
```assembly
|
||||||
leal (BP_scratch+4)(%esi), %esp
|
leal (BP_scratch+4)(%esi), %esp
|
||||||
@ -166,11 +166,11 @@ As we checked `KEEP_SEGMENTS` flag and put the correct value to the segment regi
|
|||||||
subl $1b, %ebp
|
subl $1b, %ebp
|
||||||
```
|
```
|
||||||
|
|
||||||
Here `esi` register contains address of the [boot_params](https://github.com/torvalds/linux/blob/master/arch/x86/include/uapi/asm/bootparam.h#L113) structure. `boot_params` contains special field `scratch` with offset `0x1e4`. We are getting address of the `scratch` field + 4 bytes and put it to the `esp` register (we will use it as stack for these calculations). After this we can see call instruction and `1f` label as operand of it. What does it mean `call`? It means that it pushes `ebp` value in the stack, next `esp` value, next function arguments and return address in the end. After this we pop return address from the stack into `ebp` register (`ebp` will contain return address) and subtract address of the previous label `1`.
|
Here the `esi` register contains the address of the [boot_params](https://github.com/torvalds/linux/blob/master/arch/x86/include/uapi/asm/bootparam.h#L113) structure. `boot_params` contains a special field `scratch` with offset `0x1e4`. We are getting the address of the `scratch` field + 4 bytes and puting it in the `esp` register (we will use it as stack for these calculations). After this we can see the call instruction and `1f` label as its operand. What does `call` mean? It means that it pushes the `ebp` value into the stack, then the `esp` value, then the function arguments and returns the address in the end. After this we pop return address from the stack into `ebp` register (`ebp` will contain return address) and subtract address of the previous label `1`.
|
||||||
|
|
||||||
After this we have address where we loaded in the `ebp` - `0x100000`.
|
After this we have address where we loaded in the `ebp` - `0x100000`.
|
||||||
|
|
||||||
Now we can setup the stack and verify CPU that it has support of the long mode and [SSE](http://en.wikipedia.org/wiki/Streaming_SIMD_Extensions).
|
Now we can setup the stack and verify that the CPU supports long mode and [SSE](http://en.wikipedia.org/wiki/Streaming_SIMD_Extensions).
|
||||||
|
|
||||||
Stack setup and CPU verification
|
Stack setup and CPU verification
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
@ -9,7 +9,7 @@ This is the fifth part of the `Kernel booting process` series. We saw transition
|
|||||||
Preparation before kernel decompression
|
Preparation before kernel decompression
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
We stopped right before jump on 64-bit entry point - `startup_64` which located in the [arch/x86/boot/compressed/head_64.S](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/head_64.S) source code file. We already saw the jump to the `startup_64` in the `startup_32`:
|
We stopped right before the jump on the 64-bit entry point - `startup_64` which is located in the [arch/x86/boot/compressed/head_64.S](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/head_64.S) source code file. We already saw the jump to the `startup_64` in the `startup_32`:
|
||||||
|
|
||||||
```assembly
|
```assembly
|
||||||
pushl $__KERNEL_CS
|
pushl $__KERNEL_CS
|
||||||
@ -24,7 +24,7 @@ We stopped right before jump on 64-bit entry point - `startup_64` which located
|
|||||||
lret
|
lret
|
||||||
```
|
```
|
||||||
|
|
||||||
in the previous part, `startup_64` starts to work. Since we loaded the new Global Descriptor Table and there was CPU transition in other mode (64-bit mode in our case), we can see setup of the data segments:
|
in the previous part, `startup_64` starts to work. Since we loaded the new Global Descriptor Table and there was CPU transition in other mode (64-bit mode in our case), we can see the setup of the data segments:
|
||||||
|
|
||||||
```assembly
|
```assembly
|
||||||
.code64
|
.code64
|
||||||
@ -38,9 +38,9 @@ ENTRY(startup_64)
|
|||||||
movl %eax, %gs
|
movl %eax, %gs
|
||||||
```
|
```
|
||||||
|
|
||||||
in the beginning of the `startup_64`. All segment registers besides `cs` points now to the `ds` which is `0x18` (if you don't understand why it is `0x18`, read the previous part).
|
in the beginning of the `startup_64`. All segment registers besides `cs` now point to the `ds` which is `0x18` (if you don't understand why it is `0x18`, read the previous part).
|
||||||
|
|
||||||
The next step is computation of difference between where kernel was compiled and where it was loaded:
|
The next step is computation of difference between where the kernel was compiled and where it was loaded:
|
||||||
|
|
||||||
```assembly
|
```assembly
|
||||||
#ifdef CONFIG_RELOCATABLE
|
#ifdef CONFIG_RELOCATABLE
|
||||||
@ -58,9 +58,9 @@ The next step is computation of difference between where kernel was compiled and
|
|||||||
leaq z_extract_offset(%rbp), %rbx
|
leaq z_extract_offset(%rbp), %rbx
|
||||||
```
|
```
|
||||||
|
|
||||||
`rbp` contains decompressed kernel start address and after this code executed `rbx` register will contain address where to relocate the kernel code for decompression. We already saw code like this in the `startup_32` ( you can read about it in the previous part - [Calculate relocation address](https://github.com/0xAX/linux-insides/blob/master/Booting/linux-bootstrap-4.md#calculate-relocation-address)), but we need to do this calculation again because bootloader can use 64-bit boot protocol and `startup_32` just will not be executed in this case.
|
`rbp` contains the decompressed kernel start address and after this code executes `rbx` register will contain address to relocate the kernel code for decompression. We already saw code like this in the `startup_32` ( you can read about it in the previous part - [Calculate relocation address](https://github.com/0xAX/linux-insides/blob/master/Booting/linux-bootstrap-4.md#calculate-relocation-address)), but we need to do this calculation again because the bootloader can use 64-bit boot protocol and `startup_32` just will not be executed in this case.
|
||||||
|
|
||||||
In the next step we can see setup of the stack and reset of flags register:
|
In the next step we can see setup of the stack and reset of the flags register:
|
||||||
|
|
||||||
```assembly
|
```assembly
|
||||||
leaq boot_stack_end(%rbx), %rsp
|
leaq boot_stack_end(%rbx), %rsp
|
||||||
@ -84,7 +84,7 @@ boot_stack_end:
|
|||||||
|
|
||||||
It located in the `.bss` section right before `.pgtable`. You can look at [arch/x86/boot/compressed/vmlinux.lds.S](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/vmlinux.lds.S) to find it.
|
It located in the `.bss` section right before `.pgtable`. You can look at [arch/x86/boot/compressed/vmlinux.lds.S](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/vmlinux.lds.S) to find it.
|
||||||
|
|
||||||
As we set the stack, now we can copy the compressed kernel to the address that we got above, when we calculated the relocation address of the decompressed kernel. Let's look on this code:
|
As we set the stack, now we can copy the compressed kernel to the address that we got above, when we calculated the relocation address of the decompressed kernel. Let's look at this code:
|
||||||
|
|
||||||
```assembly
|
```assembly
|
||||||
pushq %rsi
|
pushq %rsi
|
||||||
@ -98,9 +98,9 @@ As we set the stack, now we can copy the compressed kernel to the address that w
|
|||||||
popq %rsi
|
popq %rsi
|
||||||
```
|
```
|
||||||
|
|
||||||
First of all we push `rsi` to the stack. We need save value of `rsi`, because this register now stores pointer to the `boot_params` real mode structure (you must remember this structure, we filled it in the start of kernel setup). In the end of this code we'll restore pointer to the `boot_params` into `rsi` again.
|
First of all we push `rsi` to the stack. We need save the value of `rsi`, because this register now stores a pointer to the `boot_params` real mode structure (you must remember this structure, we filled it in the start of kernel setup). In the end of this code we'll restore the pointer to the `boot_params` into `rsi` again.
|
||||||
|
|
||||||
The next two `leaq` instructions calculates effective address of the `rip` and `rbx` with `_bss - 8` offset and put it to the `rsi` and `rdi`. Why we calculate this addresses? Actually compressed kernel image located between this copying code (from `startup_32` to the current code) and the decompression code. You can verify this by looking on the linker script - [arch/x86/boot/compressed/vmlinux.lds.S](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/vmlinux.lds.S):
|
The next two `leaq` instructions calculates effective address of the `rip` and `rbx` with `_bss - 8` offset and put it to the `rsi` and `rdi`. Why do we calculate these addresses? Actually the compressed kernel image is located between this copying code (from `startup_32` to the current code) and the decompression code. You can verify this by looking at the linker script - [arch/x86/boot/compressed/vmlinux.lds.S](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/vmlinux.lds.S):
|
||||||
|
|
||||||
```
|
```
|
||||||
. = 0;
|
. = 0;
|
||||||
@ -146,15 +146,15 @@ relocated:
|
|||||||
...
|
...
|
||||||
```
|
```
|
||||||
|
|
||||||
And `.rodata..compressed` contains compressed kernel image.
|
And `.rodata..compressed` contains the compressed kernel image.
|
||||||
|
|
||||||
So `rsi` will contain `rip` relative address of the `_bss - 8` and `rdi` will contain relocation relative address of the ``_bss - 8`. As we store these addresses in register, we put the address of `_bss` to the `rcx` register. As you can see in the `vmlinux.lds.S`, it located in the end of all sections with the setup/kernel code. Now we can start to copy data from `rsi` to `rdi` by 8 bytes with `movsq` instruction.
|
So `rsi` will contain `rip` relative address of the `_bss - 8` and `rdi` will contain relocation relative address of the ``_bss - 8`. As we store these addresses in register, we put the address of `_bss` to the `rcx` register. As you can see in the `vmlinux.lds.S`, it located in the end of all sections with the setup/kernel code. Now we can start to copy data from `rsi` to `rdi` by 8 bytes with `movsq` instruction.
|
||||||
|
|
||||||
Note that there is `std` instruction before data copying, it sets `DF` flag and it means that `rsi` and `rdi` will be decremeted or in other words, we will crbxopy bytes in backwards.
|
Note that there is `std` instruction before data copying, it sets `DF` flag and it means that `rsi` and `rdi` will be decremented or in other words, we will copy bytes in backwards.
|
||||||
|
|
||||||
In the end we clear `DF` flag with `cld` instruction and restore `boot_params` structure to the `rsi`.
|
In the end we clear `DF` flag with `cld` instruction and restore `boot_params` structure to the `rsi`.
|
||||||
|
|
||||||
After it we get `.text` section address and jump to it:
|
After this we get `.text` section address and jump to it:
|
||||||
|
|
||||||
```assembly
|
```assembly
|
||||||
leaq relocated(%rbx), %rax
|
leaq relocated(%rbx), %rax
|
||||||
@ -199,7 +199,7 @@ Again we save `rsi` with pointer to `boot_params` structure and call `decompress
|
|||||||
Kernel decompression
|
Kernel decompression
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
As i wrote above, `decompress_kernel` function is in the [arch/x86/boot/compressed/misc.c](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/misc.c) source code file. This function starts with the video/console initialization that we saw in the previous parts. This calls need if bootloaded used 32 or 64-bit protocols. After this we store pointers to the start of the free memory and to the end of it:
|
As I wrote above, `decompress_kernel` function is in the [arch/x86/boot/compressed/misc.c](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/misc.c) source code file. This function starts with the video/console initialization that we saw in the previous parts. This call needs to know if bootloader used 32 or 64-bit protocols. After this we store pointers to the start of the free memory and to the end of it:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
free_mem_ptr = heap;
|
free_mem_ptr = heap;
|
||||||
@ -212,7 +212,7 @@ where `heap` is the second parameter of the `decompress_kernel` function which w
|
|||||||
leaq boot_heap(%rip), %rsi
|
leaq boot_heap(%rip), %rsi
|
||||||
```
|
```
|
||||||
|
|
||||||
As you saw about `boot_heap` defined as:
|
As you saw above, `boot_heap` is defined as:
|
||||||
|
|
||||||
```assembly
|
```assembly
|
||||||
boot_heap:
|
boot_heap:
|
||||||
@ -221,7 +221,7 @@ boot_heap:
|
|||||||
|
|
||||||
where `BOOT_HEAP_SIZE` is `0x400000` if the kernel compressed with `bzip2` or `0x8000` if not.
|
where `BOOT_HEAP_SIZE` is `0x400000` if the kernel compressed with `bzip2` or `0x8000` if not.
|
||||||
|
|
||||||
In the next step we call `choose_kernel_location` function from the [arch/x86/boot/compressed/aslr.c](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/aslr.c#L298). As we can understand from the function name it chooses memory location where to decompress the kernel image. Let's look on this function.
|
In the next step we call the `choose_kernel_location` function from [arch/x86/boot/compressed/aslr.c](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/aslr.c#L298). As we can understand from the function name it chooses the memory location where the kernel image will be decompressed. Let's look at this function.
|
||||||
|
|
||||||
At the start `choose_kernel_location` tries to find `kaslr` option in the command line if `CONFIG_HIBERNATION` is set and `nokaslr` option if this configuration option `CONFIG_HIBERNATION` is not set:
|
At the start `choose_kernel_location` tries to find `kaslr` option in the command line if `CONFIG_HIBERNATION` is set and `nokaslr` option if this configuration option `CONFIG_HIBERNATION` is not set:
|
||||||
|
|
||||||
@ -246,7 +246,7 @@ out:
|
|||||||
return (unsigned char *)choice;
|
return (unsigned char *)choice;
|
||||||
```
|
```
|
||||||
|
|
||||||
which just returns the `output` parameter which we passed to the `choose_kernel_location` without any changes. Let's try to understand what is it `kaslr`. We can find information about it in the [documentation](https://github.com/torvalds/linux/blob/master/Documentation/kernel-parameters.txt):
|
which just returns the `output` parameter which we passed to the `choose_kernel_location` without any changes. Let's try to understand what `kaslr` is. We can find information about it in the [documentation](https://github.com/torvalds/linux/blob/master/Documentation/kernel-parameters.txt):
|
||||||
|
|
||||||
```
|
```
|
||||||
kaslr/nokaslr [X86]
|
kaslr/nokaslr [X86]
|
||||||
@ -258,11 +258,11 @@ kASLR is disabled by default. When kASLR is enabled,
|
|||||||
hibernation will be disabled.
|
hibernation will be disabled.
|
||||||
```
|
```
|
||||||
|
|
||||||
It means that we can pass `kaslr` option to the kernel's command line and get random address for the decompressed kernel (more about aslr you can read [here](https://en.wikipedia.org/wiki/Address_space_layout_randomization)).
|
It means that we can pass the `kaslr` option to the kernel's command line and get a random address for the decompressed kernel (you can read more about aslr [here](https://en.wikipedia.org/wiki/Address_space_layout_randomization)).
|
||||||
|
|
||||||
Let's consider the case when kernel's command line contains `kaslr` option.
|
Let's consider the case when kernel's command line contains the `kaslr` option.
|
||||||
|
|
||||||
There is the call of the `mem_avoid_init` function from the same `aslr.c` source code file. This function gets the unsafe memory regions (initrd, kernel command line and etc...). We need to know about this memory regions to not overlap them with the kernel after decompression. For example:
|
There is the call of the `mem_avoid_init` function from the same `aslr.c` source code file. This function gets the unsafe memory regions (initrd, kernel command line and etc...). We need to know about these memory regions to not overlap them with the kernel after decompression. For example:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
initrd_start = (u64)real_mode->ext_ramdisk_image << 32;
|
initrd_start = (u64)real_mode->ext_ramdisk_image << 32;
|
||||||
@ -299,7 +299,7 @@ Offset Proto Name Meaning
|
|||||||
...
|
...
|
||||||
```
|
```
|
||||||
|
|
||||||
So we're taking `ext_ramdisk_image` and `ext_ramdisk_size`, shifting they left on 32 (now they will contain low 32-bits in the high 32-bit bits) and getting start address of the `initrd` and size of it. After this we store these values in the `mem_avoid` array which defined as:
|
So we're taking `ext_ramdisk_image` and `ext_ramdisk_size`, shifting them left on 32 (now they will contain low 32-bits in the high 32-bit bits) and getting start address of the `initrd` and size of it. After this we store these values in the `mem_avoid` array which is defined as:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
#define MEM_AVOID_MAX 5
|
#define MEM_AVOID_MAX 5
|
||||||
@ -315,7 +315,7 @@ struct mem_vector {
|
|||||||
};
|
};
|
||||||
```
|
```
|
||||||
|
|
||||||
The next step after we collected all unsafe memory regions in the `mem_avoid` array will be search of the random address which does not overlap with the unsafe regions with the `find_random_addr` function.
|
The next step after we collect all unsafe memory regions in the `mem_avoid` array will be searching for the random address which does not overlap with the unsafe regions with the `find_random_addr` function.
|
||||||
|
|
||||||
First of all we can see align of the output address in the `find_random_addr` function:
|
First of all we can see align of the output address in the `find_random_addr` function:
|
||||||
|
|
||||||
@ -323,7 +323,7 @@ First of all we can see align of the output address in the `find_random_addr` fu
|
|||||||
minimum = ALIGN(minimum, CONFIG_PHYSICAL_ALIGN);
|
minimum = ALIGN(minimum, CONFIG_PHYSICAL_ALIGN);
|
||||||
```
|
```
|
||||||
|
|
||||||
you can remember `CONFIG_PHYSICAL_ALIGN` configuration option from the previous part. This option provides the value to which kernel should be aligned and it is `0x200000` by default. After that we got aligned output address, we go through the memory and collect regions which are good for decompressed kernel image:
|
you can remember `CONFIG_PHYSICAL_ALIGN` configuration option from the previous part. This option provides the value to which kernel should be aligned and it is `0x200000` by default. Once we have the aligned output address, we go through the memory and collect regions which are good for decompressed kernel image:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
for (i = 0; i < real_mode->e820_entries; i++) {
|
for (i = 0; i < real_mode->e820_entries; i++) {
|
||||||
@ -331,7 +331,7 @@ for (i = 0; i < real_mode->e820_entries; i++) {
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
You can remember that we collected `e820_entries` in the second part of the [Kernel booting process part 2](https://github.com/0xAX/linux-insides/blob/master/Booting/linux-bootstrap-2.md#memory-detection).
|
Recall that we collected `e820_entries` in the second part of the [Kernel booting process part 2](https://github.com/0xAX/linux-insides/blob/master/Booting/linux-bootstrap-2.md#memory-detection).
|
||||||
|
|
||||||
First of all `process_e820_entry` function does some checks that e820 memory region is not non-RAM, that the start address of the memory region is not bigger than Maximum allowed `aslr` offset and that memory region is not less than value of kernel alignment:
|
First of all `process_e820_entry` function does some checks that e820 memory region is not non-RAM, that the start address of the memory region is not bigger than Maximum allowed `aslr` offset and that memory region is not less than value of kernel alignment:
|
||||||
|
|
||||||
@ -355,7 +355,7 @@ region.start = entry->addr;
|
|||||||
region.size = entry->size;
|
region.size = entry->size;
|
||||||
```
|
```
|
||||||
|
|
||||||
As we store these values, we align the `region.start` as we did it in the `find_random_addr` function and check that we didn't get address that bigger than original memory region:
|
As we store these values, we align the `region.start` as we did it in the `find_random_addr` function and check that we didn't get an address that is bigger than original memory region:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
region.start = ALIGN(region.start, CONFIG_PHYSICAL_ALIGN);
|
region.start = ALIGN(region.start, CONFIG_PHYSICAL_ALIGN);
|
||||||
@ -364,7 +364,7 @@ if (region.start > entry->addr + entry->size)
|
|||||||
return;
|
return;
|
||||||
```
|
```
|
||||||
|
|
||||||
Next we get difference between the original address and aligned and check that if the last address in the memory region is bigger than `CONFIG_RANDOMIZE_BASE_MAX_OFFSET`, we reduce the memory region size that end of kernel image will be less than maximum `aslr` offset:
|
Next we get the difference between the original address and aligned and check that if the last address in the memory region is bigger than `CONFIG_RANDOMIZE_BASE_MAX_OFFSET`, we reduce the memory region size so that the end of the kernel image will be less than the maximum `aslr` offset:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
region.size -= region.start - entry->addr;
|
region.size -= region.start - entry->addr;
|
||||||
@ -373,7 +373,7 @@ if (region.start + region.size > CONFIG_RANDOMIZE_BASE_MAX_OFFSET)
|
|||||||
region.size = CONFIG_RANDOMIZE_BASE_MAX_OFFSET - region.start;
|
region.size = CONFIG_RANDOMIZE_BASE_MAX_OFFSET - region.start;
|
||||||
```
|
```
|
||||||
|
|
||||||
In the end we go through the all unsafe memory regions and check that this region does not overlap unsafe ares with kernel command line, initrd and etc...:
|
In the end we go through all unsafe memory regions and check that each region does not overlap unsafe ares with kernel command line, initrd and etc...:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
for (img.start = region.start, img.size = image_size ;
|
for (img.start = region.start, img.size = image_size ;
|
||||||
@ -385,13 +385,13 @@ for (img.start = region.start, img.size = image_size ;
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
If memory region does not overlap unsafe regions we call `slots_append` function with the start address of the region. `slots_append` function just collects start addresses of memory regions to the `slots` array:
|
If the memory region does not overlap unsafe regions we call the `slots_append` function with the start address of the region. `slots_append` function just collects start addresses of memory regions to the `slots` array:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
slots[slot_max++] = addr;
|
slots[slot_max++] = addr;
|
||||||
```
|
```
|
||||||
|
|
||||||
which defined as:
|
which is defined as:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
static unsigned long slots[CONFIG_RANDOMIZE_BASE_MAX_OFFSET /
|
static unsigned long slots[CONFIG_RANDOMIZE_BASE_MAX_OFFSET /
|
||||||
@ -399,7 +399,7 @@ static unsigned long slots[CONFIG_RANDOMIZE_BASE_MAX_OFFSET /
|
|||||||
static unsigned long slot_max;
|
static unsigned long slot_max;
|
||||||
```
|
```
|
||||||
|
|
||||||
After `process_e820_entry` will be executed, we will have array of the addresses which are safe for the decompressed kernel. Next we call `slots_fetch_random` function for getting random item from this array:
|
After `process_e820_entry` will be executed, we will have an array of the addresses which are safe for the decompressed kernel. Next we call `slots_fetch_random` function for getting random item from this array:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
if (slot_max == 0)
|
if (slot_max == 0)
|
||||||
@ -408,9 +408,9 @@ if (slot_max == 0)
|
|||||||
return slots[get_random_long() % slot_max];
|
return slots[get_random_long() % slot_max];
|
||||||
```
|
```
|
||||||
|
|
||||||
where `get_random_long` function checks different CPU flags as `X86_FEATURE_RDRAND` or `X86_FEATURE_TSC` and chooses method for getting random number (it can be obtain with RDRAND instruction, Time stamp counter, programmable interval timer and etc...). After that we got random address execution of the `choose_kernel_location` is finished.
|
where `get_random_long` function checks different CPU flags as `X86_FEATURE_RDRAND` or `X86_FEATURE_TSC` and chooses method for getting random number (it can be obtain with RDRAND instruction, Time stamp counter, programmable interval timer and etc...). After retrieving the random address execution of the `choose_kernel_location` is finished.
|
||||||
|
|
||||||
Now let's back to the [misc.c](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/misc.c#L404). After we got address for the kernel image, there need to do some checks to be sure that gotten random address is correctly aligned and address is not wrong.
|
Now let's back to the [misc.c](https://github.com/torvalds/linux/blob/master/arch/x86/boot/compressed/misc.c#L404). After getting the address for the kernel image, there need to be some checks to be sure that the retrieved random address is correctly aligned and address is not wrong.
|
||||||
|
|
||||||
After all these checks will see the familiar message:
|
After all these checks will see the familiar message:
|
||||||
|
|
||||||
@ -418,7 +418,7 @@ After all these checks will see the familiar message:
|
|||||||
Decompressing Linux...
|
Decompressing Linux...
|
||||||
```
|
```
|
||||||
|
|
||||||
and call `decompress` function which will decompress the kernel. `decompress` function depends on what decompression algorithm was chosen during kernel compilartion:
|
and call `decompress` function which will decompress the kernel. `decompress` function depends on what decompression algorithm was chosen during kernel compilation:
|
||||||
|
|
||||||
```C
|
```C
|
||||||
#ifdef CONFIG_KERNEL_GZIP
|
#ifdef CONFIG_KERNEL_GZIP
|
||||||
@ -446,7 +446,7 @@ and call `decompress` function which will decompress the kernel. `decompress` fu
|
|||||||
#endif
|
#endif
|
||||||
```
|
```
|
||||||
|
|
||||||
After kernel will be decompressed, the last function `handle_relocations` will relocate the kernel to the address that we got from `choose_kernel_location`. After that kernel relocated we return from the `decompress_kernel` to the `head_64.S`. The address of the kernel will be in the `rax` register and we jump on it:
|
After kernel will be decompressed, the last function `handle_relocations` will relocate the kernel to the address that we got from `choose_kernel_location`. After the kernel is relocated we return from the `decompress_kernel` to `head_64.S`. The address of the kernel will be in the `rax` register and we jump to it:
|
||||||
|
|
||||||
```assembly
|
```assembly
|
||||||
jmp *%rax
|
jmp *%rax
|
||||||
|
@ -378,7 +378,7 @@ Ok, we already passed the main theme of this part which is `RCU` initialization,
|
|||||||
|
|
||||||
After we initilized `RCU`, the next step which you can see in the [init/main.c](https://github.com/torvalds/linux/blob/master/init/main.c) is the - `trace_init` function. As you can understand from its name, this function initialize [tracing](http://en.wikipedia.org/wiki/Tracing_%28software%29) subsystem. More about linux kernel trace system you can read - [here](http://elinux.org/Kernel_Trace_Systems).
|
After we initilized `RCU`, the next step which you can see in the [init/main.c](https://github.com/torvalds/linux/blob/master/init/main.c) is the - `trace_init` function. As you can understand from its name, this function initialize [tracing](http://en.wikipedia.org/wiki/Tracing_%28software%29) subsystem. More about linux kernel trace system you can read - [here](http://elinux.org/Kernel_Trace_Systems).
|
||||||
|
|
||||||
After the `trace_init`, we can see the call of the `radix_tree_init`. If you are familar with the different data structures, you can understand from the name of this function that it initializes kernel implementation of the [Radix tree](http://en.wikipedia.org/wiki/Radix_tree). This function defined in the [lib/radix-tree.c](https://github.com/torvalds/linux/blob/master/lib/radix-tree.c) and more about it you can read in the part about [Radix tree](http://0xax.gitbooks.io/linux-insides/content/DataStructures/radix-tree.md).
|
After the `trace_init`, we can see the call of the `radix_tree_init`. If you are familar with the different data structures, you can understand from the name of this function that it initializes kernel implementation of the [Radix tree](http://en.wikipedia.org/wiki/Radix_tree). This function defined in the [lib/radix-tree.c](https://github.com/torvalds/linux/blob/master/lib/radix-tree.c) and more about it you can read in the part about [Radix tree](https://0xax.gitbooks.io/linux-insides/content/DataStructures/radix-tree.html).
|
||||||
|
|
||||||
In the next step we can see the functions which are related to the `interrupts handling` subsystem, they are:
|
In the next step we can see the functions which are related to the `interrupts handling` subsystem, they are:
|
||||||
|
|
||||||
|
@ -34,7 +34,8 @@
|
|||||||
* [vsyscall and vDSO](SysCall/syscall-3.md)
|
* [vsyscall and vDSO](SysCall/syscall-3.md)
|
||||||
* [How the Linux kernel runs a program](SysCall/syscall-4.md)
|
* [How the Linux kernel runs a program](SysCall/syscall-4.md)
|
||||||
* [Timers and time management](Timers/README.md)
|
* [Timers and time management](Timers/README.md)
|
||||||
* [Introduction](Timers/timers-1.md)
|
* [Introduction](Timers/timers-1.md)
|
||||||
|
* [Clocksource framework](Timers/timers-2.md)
|
||||||
* [Memory management](mm/README.md)
|
* [Memory management](mm/README.md)
|
||||||
* [Memblock](mm/linux-mm-1.md)
|
* [Memblock](mm/linux-mm-1.md)
|
||||||
* [Fixmaps and ioremap](mm/linux-mm-2.md)
|
* [Fixmaps and ioremap](mm/linux-mm-2.md)
|
||||||
|
@ -4,14 +4,14 @@ System calls in the Linux kernel. Part 1.
|
|||||||
Introduction
|
Introduction
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
This post opens up a new chapter in [linux-insides](http://0xax.gitbooks.io/linux-insides/content/) book and as you may understand from the title, this chapter will be devoted to the [System call](https://en.wikipedia.org/wiki/System_call) concept in the Linux kernel. The choice of topic for this chapter is not accidental. In the previous [chapter](http://0xax.gitbooks.io/linux-insides/content/interrupts/index.html) we saw interrupts and interrupt handling. The concept of system calls is very similar to that of interrupts. This is because the most common way to implement system calls is as software interrupts. We will see many different aspects that are related to the system call concept. For example, we will learn what's happening when a system call occurs from userspace, we will see an implementation of a couple system call handlers in the Linux kernel, [VDSO](https://en.wikipedia.org/wiki/VDSO) and [vsyscall](https://lwn.net/Articles/446528/) concepts and many many more.
|
This post opens up a new chapter in [linux-insides](http://0xax.gitbooks.io/linux-insides/content/) book, and as you may understand from the title, this chapter will be devoted to the [System call](https://en.wikipedia.org/wiki/System_call) concept in the Linux kernel. The choice of topic for this chapter is not accidental. In the previous [chapter](http://0xax.gitbooks.io/linux-insides/content/interrupts/index.html) we saw interrupts and interrupt handling. The concept of system calls is very similar to that of interrupts. This is because the most common way to implement system calls is as software interrupts. We will see many different aspects that are related to the system call concept. For example, we will learn what's happening when a system call occurs from userspace. We will see an implementation of a couple system call handlers in the Linux kernel, [VDSO](https://en.wikipedia.org/wiki/VDSO) and [vsyscall](https://lwn.net/Articles/446528/) concepts and many many more.
|
||||||
|
|
||||||
Before we start to dive into the implementation of the system calls related stuff in the Linux kernel source code, it is good to know some theory about system calls. Let's do it in the following paragraph.
|
Before we dive into Linux system call implementation, it is good to know some theory about system calls. Let's do it in the following paragraph.
|
||||||
|
|
||||||
System call. What is it?
|
System call. What is it?
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
A system call is just a userspace request of a kernel service. Yes, the operating system kernel provides many services. When your program wants to write to or read from a file, start to listen for connections on a [socket](https://en.wikipedia.org/wiki/Network_socket), delete or create directory, or even to finish its work, a program uses a system call. In another words, a system call is just a [C](https://en.wikipedia.org/wiki/C_%28programming_language%29) function that is placed in the kernel space and an user program can ask kernel to do something via this function.
|
A system call is just a userspace request of a kernel service. Yes, the operating system kernel provides many services. When your program wants to write to or read from a file, start to listen for connections on a [socket](https://en.wikipedia.org/wiki/Network_socket), delete or create directory, or even to finish its work, a program uses a system call. In another words, a system call is just a [C](https://en.wikipedia.org/wiki/C_%28programming_language%29) kernel space function that user space progams call to handle some request.
|
||||||
|
|
||||||
The Linux kernel provides a set of these functions and each architecture provides its own set. For example: the [x86_64](https://en.wikipedia.org/wiki/X86-64) provides [322](https://github.com/torvalds/linux/blob/master/arch/x86/entry/syscalls/syscall_64.tbl) system calls and the [x86](https://en.wikipedia.org/wiki/X86) provides [358](https://github.com/torvalds/linux/blob/master/arch/x86/entry/syscalls/syscall_32.tbl) different system calls. Ok, a system call is just a function. Let's look on a simple `Hello world` example that's written in the assembly programming language:
|
The Linux kernel provides a set of these functions and each architecture provides its own set. For example: the [x86_64](https://en.wikipedia.org/wiki/X86-64) provides [322](https://github.com/torvalds/linux/blob/master/arch/x86/entry/syscalls/syscall_64.tbl) system calls and the [x86](https://en.wikipedia.org/wiki/X86) provides [358](https://github.com/torvalds/linux/blob/master/arch/x86/entry/syscalls/syscall_32.tbl) different system calls. Ok, a system call is just a function. Let's look on a simple `Hello world` example that's written in the assembly programming language:
|
||||||
|
|
||||||
|
@ -3,3 +3,4 @@
|
|||||||
This chapter describes timers and time management related concepts in the linux kernel.
|
This chapter describes timers and time management related concepts in the linux kernel.
|
||||||
|
|
||||||
* [Introduction](http://0xax.gitbooks.io/linux-insides/content/Timers/timers-1.html) - this part is introduction to the timers in the Linux kernel.
|
* [Introduction](http://0xax.gitbooks.io/linux-insides/content/Timers/timers-1.html) - this part is introduction to the timers in the Linux kernel.
|
||||||
|
* [Introduction to the clocksource framework](https://github.com/0xAX/linux-insides/blob/master/Timers/timers-2.md) - this part describes `clocksource` framework in the Linux kernel.
|
||||||
|
451
Timers/timers-2.md
Normal file
451
Timers/timers-2.md
Normal file
@ -0,0 +1,451 @@
|
|||||||
|
Timers in the Linux kernel. Part 2.
|
||||||
|
================================================================================
|
||||||
|
|
||||||
|
Introduction to the `clocksource` framework
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
The previous [part](https://0xax.gitbooks.io/linux-insides/content/Timers/timers-1.html) was the first part in the current [chapter](https://0xax.gitbooks.io/linux-insides/content/Timers/index.html) that describes timers and time management related stuff in the Linux kernel. We got acquainted with two concepts in the previous part:
|
||||||
|
|
||||||
|
* `jiffies`
|
||||||
|
* `clocksource`
|
||||||
|
|
||||||
|
The first is the global variable that defined in the [include/linux/jiffies.h](https://github.com/torvalds/linux/blob/master/include/linux/jiffies.h) header file and represents counter that incremented during each timer interrupt. So if we can access this global variable and we know timer interrupt rate we can convert `jiffies` to the human time units. As we already know the timer interrupt rate represented by the compile-time constant that is called `HZ` in the Linux kernel. The value of the `HZ` is equal to the value of the `CONFIG_HZ` kernel configuration option and if we will look in the [arch/x86/configs/x86_64_defconfig](https://github.com/torvalds/linux/blob/master/arch/x86/configs/x86_64_defconfig) kernel configuration file, we will see that:
|
||||||
|
|
||||||
|
```
|
||||||
|
CONFIG_HZ_1000=y
|
||||||
|
```
|
||||||
|
|
||||||
|
kernel configuration option is set. This means that value of the `CONFIG_HZ` will be `1000` by default for the [x86_64](https://en.wikipedia.org/wiki/X86-64) architecture. So, if we divide values of `jiffies` on the value of the `HZ`:
|
||||||
|
|
||||||
|
```
|
||||||
|
jiffies / HZ
|
||||||
|
```
|
||||||
|
|
||||||
|
we will get amount of seconds that elapsed since the beginning of the moment when the Linux kernel started to work or in other words we will get system [uptime](https://en.wikipedia.org/wiki/Uptime). Since the `HZ` represents amount of the timer interrupts in a second, we can set a value for some time in the future. For example:
|
||||||
|
|
||||||
|
```C
|
||||||
|
/* one minute from now */
|
||||||
|
unsigned long later = jiffies + 60*HZ;
|
||||||
|
|
||||||
|
/* five minutes from now */
|
||||||
|
unsigned long later = jiffies + 5*60*HZ;
|
||||||
|
```
|
||||||
|
|
||||||
|
This is a very common practice in the Linux kernel. For example, if you will look in the [arch/x86/kernel/smpboot.c](https://github.com/torvalds/linux/blob/master/arch/x86/kernel/smpboot.c) source code file, you will find the `do_boot_cpu` function. This function boots all processors besides bootstrap processor. You can find a piece of code that waits for ten seconds for a response from application processor:
|
||||||
|
|
||||||
|
```C
|
||||||
|
if (!boot_error) {
|
||||||
|
timeout = jiffies + 10*HZ;
|
||||||
|
while (time_before(jiffies, timeout)) {
|
||||||
|
...
|
||||||
|
...
|
||||||
|
...
|
||||||
|
udelay(100);
|
||||||
|
}
|
||||||
|
...
|
||||||
|
...
|
||||||
|
...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
We assign `jiffies + 10*HZ` value to the `timeout` variable here. As I think you already understood, this will mean ten seconds timeout. After this we are entering to the loop that we use `time_before` macro to compare current `jiffies` value and our timeout.
|
||||||
|
|
||||||
|
Or for example if we will look in the [sound/isa/sscape.c](https://github.com/torvalds/linux/blob/master/sound/isa/sscape) source code file which represents driver for the [Ensoniq Soundscape Elite](https://en.wikipedia.org/wiki/Ensoniq_Soundscape_Elite) sound card, we will see the `obp_startup_ack` function that waits given timeout for On-Board Processor to return its start-up acknowledgement sequence:
|
||||||
|
|
||||||
|
```C
|
||||||
|
static int obp_startup_ack(struct soundscape *s, unsigned timeout)
|
||||||
|
{
|
||||||
|
unsigned long end_time = jiffies + msecs_to_jiffies(timeout);
|
||||||
|
|
||||||
|
do {
|
||||||
|
...
|
||||||
|
...
|
||||||
|
...
|
||||||
|
x = host_read_unsafe(s->io_base);
|
||||||
|
...
|
||||||
|
...
|
||||||
|
...
|
||||||
|
if (x == 0xfe || x == 0xff)
|
||||||
|
return 1;
|
||||||
|
msleep(10);
|
||||||
|
} while (time_before(jiffies, end_time));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
So, you can find that `jiffies` variable is very widely used in the Linux kernel [code](http://lxr.free-electrons.com/ident?i=jiffies). As I already wrote, we met yet another new time management related concept in the previous part - `clocksource`. In the previous part we just saw a little descrption of this concept and saw API for a clock source registration. Let's take a closer look at this concept in this part
|
||||||
|
|
||||||
|
Introduction to `clocksource`
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
The `clocksource` concept represents generic API for clock sources management in the Linux kernel. Why do we need separate framework for this? Let's go back to the beginning. The `time` concept is fundamental concept in the Linux kernel and other operating system kernels. And the timekeeping is an one one of the necessities to use this concept. For example Linux kernel must know and update the time elapsed since system startup, it must determine how long the current process has been running for every processor and many many more. Where the Linux kernel can get information about time? First of all it is Real Time Clock or [RTC](https://en.wikipedia.org/wiki/Real-time_clock) that represents by the a nonvolatile device. You can find a set of architecture-independend real time clock drivers in the Linux kernel in the [drivers/rtc](https://github.com/torvalds/linux/tree/master/drivers/rtc) directory. Besides this, each architecture can provide a driver for the architecture-dependend real time clock, for example - `CMOS/RTC` - [arch/x86/kernel/rtc.c](https://github.com/torvalds/linux/blob/master/arch/x86/kernel/rtc.c) for the [x86](https://en.wikipedia.org/wiki/X86) architecture. The second is system timer - timer that excites [interrupts](https://en.wikipedia.org/wiki/Interrupt) with a periodic rate. For example, for [IBM PC](https://en.wikipedia.org/wiki/IBM_Personal_Computer) compatibles it was - [programmable interval timer](https://en.wikipedia.org/wiki/Programmable_interval_timer).
|
||||||
|
|
||||||
|
We already know that for timekeeping purposes we can use `jiffies` in the Linux kernel. The `jiffies` can be considered as read only global variable which is updated with `HZ` frequency. We know that the `HZ` is a compile-time kernel parameter whose reasonable range is from `100` to `1000` [Hz](https://en.wikipedia.org/wiki/Hertz). So, it is guaranteed to have an interface for time measurement with `1` - `10` milliseconds resolution. Besides standard `jiffies`, we saw the `refined_jiffies` clock source in the previous part that is based on the `i8253/i8254` [programmable interval timer](https://en.wikipedia.org/wiki/Programmable_interval_timer) tick rate which is almost `1193182` hertz. So we can get something about `1` microsecond resolution with the `refined_jiffies`. In this time, [nanoseconds](https://en.wikipedia.org/wiki/Nanosecond) are the favorite choice for the time value units of the given clock source.
|
||||||
|
|
||||||
|
The availability of more precise techniques for time intervals measurement is hardware-dependend. We just knew a little about `x86` dependend timers hardware. But each architecture provides own timers hardware. Earlier each architecture had own implementation for this purpose. Solution of this problem is an abstraction layer and associated API in a common code framework for managing various clock sources and independent of the timer interrupt. This commn code framework became - `clocksource` framework.
|
||||||
|
|
||||||
|
Generic timeofday and clock source management framework moved lot of timekeeping code into architecture independent portion of code, with architecture portion reduced to defining and managing low level hardware pieces of clocksources. A large amount of funds to measure the time interval on different architectures with different hardware is a big complexity. Implementation of the each clock releated service is strongly associated with an individual hardware device and as you can understand, it results in similar implementations for different architectures.
|
||||||
|
|
||||||
|
Within this framework, each clock source is required to maintain a representation of time as a monotonically increasing value. As we can see in the Linux kernel code, nanoseconds are the favorite choice for the time value units of a clock source in this time. One of the main point of the clock source framework is to allow an user to select clock source among a range of available hardware devices supporting clock functions when configuring the system and selecting, accessing and scaling different clock sources.
|
||||||
|
|
||||||
|
The clocksource structure
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
The fundamental of the `clocksource` framework is the `clocksource` structure that defined in the [include/linux/clocksource.h](https://github.com/torvalds/linux/blob/master/include/linux/clocksource.h) header file. We already saw some fields that are provided by the `clocksource` strucutre in the previous [part](https://0xax.gitbooks.io/linux-insides/content/Timers/timers-1.html). Let's look on the full definition of this structure and try to describe all of its fields:
|
||||||
|
|
||||||
|
```C
|
||||||
|
struct clocksource {
|
||||||
|
cycle_t (*read)(struct clocksource *cs);
|
||||||
|
cycle_t mask;
|
||||||
|
u32 mult;
|
||||||
|
u32 shift;
|
||||||
|
u64 max_idle_ns;
|
||||||
|
u32 maxadj;
|
||||||
|
#ifdef CONFIG_ARCH_CLOCKSOURCE_DATA
|
||||||
|
struct arch_clocksource_data archdata;
|
||||||
|
#endif
|
||||||
|
u64 max_cycles;
|
||||||
|
const char *name;
|
||||||
|
struct list_head list;
|
||||||
|
int rating;
|
||||||
|
int (*enable)(struct clocksource *cs);
|
||||||
|
void (*disable)(struct clocksource *cs);
|
||||||
|
unsigned long flags;
|
||||||
|
void (*suspend)(struct clocksource *cs);
|
||||||
|
void (*resume)(struct clocksource *cs);
|
||||||
|
#ifdef CONFIG_CLOCKSOURCE_WATCHDOG
|
||||||
|
struct list_head wd_list;
|
||||||
|
cycle_t cs_last;
|
||||||
|
cycle_t wd_last;
|
||||||
|
#endif
|
||||||
|
struct module *owner;
|
||||||
|
} ____cacheline_aligned;
|
||||||
|
```
|
||||||
|
|
||||||
|
We already saw the first field of the `clocksource` structure in the previous part - it is pointer to the `read` function that returns best conter selected by the clocksource framework. For example we use `jiffies_read` function to read `jiffies` value:
|
||||||
|
|
||||||
|
```C
|
||||||
|
static struct clocksource clocksource_jiffies = {
|
||||||
|
...
|
||||||
|
.read = jiffies_read,
|
||||||
|
...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
where `jiffies_read` just returns:
|
||||||
|
|
||||||
|
```C
|
||||||
|
static cycle_t jiffies_read(struct clocksource *cs)
|
||||||
|
{
|
||||||
|
return (cycle_t) jiffies;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Or the `read_tsc` function:
|
||||||
|
|
||||||
|
```C
|
||||||
|
static struct clocksource clocksource_tsc = {
|
||||||
|
...
|
||||||
|
.read = read_tsc,
|
||||||
|
...
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
for the [time stamp counter](https://en.wikipedia.org/wiki/Time_Stamp_Counter) reading.
|
||||||
|
|
||||||
|
The next field is `mask` that allows to ensure that subtraction between counters values from non `64 bit` counters do not need special overflow logic. After the `mask` field, we can see two fields: `mult` and `shift`. These are the fields that are base of mathematical functions that are provide ability to convert time values specific to each clock source. In other words these two fields help us to convert an abstract machine time units of a counter to nanoseconds.
|
||||||
|
|
||||||
|
After these two fields we can see the `64` bits `max_idle_ns` field represents max idle time permitted by the clocksource in nanoseconds. We need in this field for the Linux kernel with enabled `CONFIG_NO_HZ` kernel configuration option. This kernel configuration option enables the Linux kernel to run without a regular timer tick (we will see full explanation of this in other part). The problem that dynamic tick allows the kernel to sleep for periods longer than a single tick, moreover sleep time could be unlimited. The `max_idle_ns` field represents this sleeping limit.
|
||||||
|
|
||||||
|
The next field after the `max_idle_ns` is the `maxadj` field which is the maximum adjustment value to `mult`. The main formula by which we convert cycles to the nanoseconds:
|
||||||
|
|
||||||
|
```C
|
||||||
|
((u64) cycles * mult) >> shift;
|
||||||
|
```
|
||||||
|
|
||||||
|
is not `100%` accurate. Instead the number is taken as close as possible to a nanosecond and `maxadj` helps to correct this and allows clocksource API to avoid `mult` values that might overflow when adjusted. The next four fields are pointers to the function:
|
||||||
|
|
||||||
|
* `enable` - optional function to enable clocksource;
|
||||||
|
* `disable` - optional function to disable clocksource;
|
||||||
|
* `suspend` - suspend function for the clocksource;
|
||||||
|
* `resume` - resume function for the clocksource;
|
||||||
|
|
||||||
|
The next field is the `max_cycles` and as we can understand from its name, this field represents maximum cycle value before potential overflow. And the last field is `owner` represents reference to a kernel [module](https://en.wikipedia.org/wiki/Loadable_kernel_module) that is owner of a clocksource. This is all. We just went through all the standard fields of the `clocksource` structure. But you can noted that we missed some fields of the `clocksource` structure. We can divide all of missed field on two types: Fields of the first type are already known for us. For example, they are `name` field that represents name of a `clocksource`, the `rating` field that helps to the Linux kernel to select the best clocksource and etc. The second type, fields which are dependent from the different Linux kernel configuration options. Let's look on these fields.
|
||||||
|
|
||||||
|
The first field is the `archdata`. This field has `arch_clocksource_data` type and depends on the `CONFIG_ARCH_CLOCKSOURCE_DATA` kernel configuration option. This field is actual only for the [x86](https://en.wikipedia.org/wiki/X86) and [IA64](https://en.wikipedia.org/wiki/IA-64) architectures for this moment. And again, as we can understand from the field's name, it represents architecture-specific data for a clock source. For example, it represents `vDSO` clock mode:
|
||||||
|
|
||||||
|
```C
|
||||||
|
struct arch_clocksource_data {
|
||||||
|
int vclock_mode;
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
for the `x86` architectures. Where the `vDSO` clock mode can be one of the:
|
||||||
|
|
||||||
|
```C
|
||||||
|
#define VCLOCK_NONE 0
|
||||||
|
#define VCLOCK_TSC 1
|
||||||
|
#define VCLOCK_HPET 2
|
||||||
|
#define VCLOCK_PVCLOCK 3
|
||||||
|
```
|
||||||
|
|
||||||
|
The last three fields are `wd_list`, `cs_last` and the `wd_last` depends on the `CONFIG_CLOCKSOURCE_WATCHDOG` kernel configuration option. First of all let's try to understand what is it `whatchdog`. In a simple words, watchdog is a timer that is used for detection of the computer malfunctions and recovering from it. All of these three fields contain watchdog related data that is used by the `clocksource` framework. If we will grep the Linux kernel source code, we will see that only [arch/x86/KConfig](https://github.com/torvalds/linux/blob/master/arch/x86/Kconfig#L54) kernel configuration file contains the `CONFIG_CLOCKSOURCE_WATCHDOG` kernel configuration option. So, why do `x86` and `x86_64` need in [watchdog](https://en.wikipedia.org/wiki/Watchdog_timer)? You already may know that all `x86` processors has special 64-bit register - [time stamp counter](https://en.wikipedia.org/wiki/Time_Stamp_Counter). This register contains number of [cycles](https://en.wikipedia.org/wiki/Clock_rate) since the reset. Sometimes the time stamp counter needs to be verified against another clock source. We will not see initialization of the `watchdog` timer in this part, before this we must learn more about timers.
|
||||||
|
|
||||||
|
That's all. From this moment we know all fields of the `clocksource` structure. This knowledge will help us to learn internals of the `clocksource` framework.
|
||||||
|
|
||||||
|
New clock source registration
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
We saw only one function from the `clocksource` framework in the previous [part](https://0xax.gitbooks.io/linux-insides/content/Timers/timers-1.html). This function was - `__clocksource_register`. This function defined in the [include/linux/clocksource.h](https://github.com/torvalds/linux/tree/master/include/linux/clocksource.h) header file and as we can understand from the function's name, main point of this function is to register new clocksource. If we will look on the implementation of the `__clocksource_register` function, we will see that it just makes call of the `__clocksource_register_scale` function and returns its result:
|
||||||
|
|
||||||
|
```C
|
||||||
|
static inline int __clocksource_register(struct clocksource *cs)
|
||||||
|
{
|
||||||
|
return __clocksource_register_scale(cs, 1, 0);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Before we will see implementation of the `__clocksource_register_scale` function, we can see that `clocksource` provides additional API for a new clock source registration:
|
||||||
|
|
||||||
|
```C
|
||||||
|
static inline int clocksource_register_hz(struct clocksource *cs, u32 hz)
|
||||||
|
{
|
||||||
|
return __clocksource_register_scale(cs, 1, hz);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int clocksource_register_khz(struct clocksource *cs, u32 khz)
|
||||||
|
{
|
||||||
|
return __clocksource_register_scale(cs, 1000, khz);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
And all of these functions do the same. They return value of the `__clocksource_register_scale` function but with diffferent set of parameters. The `__clocksource_register_scale` function defined in the [kernel/time/clocksource.c](https://github.com/torvalds/linux/tree/master/kernel/time/clocksource.c) source code file. To understand difference between these functions, let's look on the parameters of the `clocksource_register_khz` function. As we can see, this function takes three parameters:
|
||||||
|
|
||||||
|
* `cs` - clocksource to be installed;
|
||||||
|
* `scale` - scale factor of a clock source. In other words, if we will multiply value of this parameter on frequency, we will get `hz` of a clocksource;
|
||||||
|
* `freq` - clock source frequency divided by scale.
|
||||||
|
|
||||||
|
Now let's look on the implementation of the `__clocksource_register_scale` function:
|
||||||
|
|
||||||
|
```C
|
||||||
|
int __clocksource_register_scale(struct clocksource *cs, u32 scale, u32 freq)
|
||||||
|
{
|
||||||
|
__clocksource_update_freq_scale(cs, scale, freq);
|
||||||
|
mutex_lock(&clocksource_mutex);
|
||||||
|
clocksource_enqueue(cs);
|
||||||
|
clocksource_enqueue_watchdog(cs);
|
||||||
|
clocksource_select();
|
||||||
|
mutex_unlock(&clocksource_mutex);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
First of all we can see that the `__clocksource_register_scale` function starts from the call of the `__clocksource_update_freq_scale` function that defined in the same source code file and updates given clock source with the new frequency. Let's look on the implementation of this function. In the first step we need to check given frequency and if it was not passed as `zero`, we need to calculate `mult` and `shift` parameters for the given clock source. Why do we need to check value of the `frequency`? Actually it can be zero. if you attentively looked on the implementation of the `__clocksource_register` function, you may have noticed that we passed `frequency` as `0`. We will do it only for some clock sources that have self defined `mult` and `shift` parameters. Look in the previous [part](https://0xax.gitbooks.io/linux-insides/content/Timers/timers-1.html) and you will see that we saw calculation of the `mult` and `shift` for `jiffies`. The `__clocksource_update_freq_scale` function will do it for us for other clock sources.
|
||||||
|
|
||||||
|
So in the start of the `__clocksource_update_freq_scale` function we check the value of the `frequency` parameter and if is not zero we need to calculate `mult` and `shift` for the given clock source. Let's look on the `mult` and `shift` calculation:
|
||||||
|
|
||||||
|
```C
|
||||||
|
void __clocksource_update_freq_scale(struct clocksource *cs, u32 scale, u32 freq)
|
||||||
|
{
|
||||||
|
u64 sec;
|
||||||
|
|
||||||
|
if (freq) {
|
||||||
|
sec = cs->mask;
|
||||||
|
do_div(sec, freq);
|
||||||
|
do_div(sec, scale);
|
||||||
|
|
||||||
|
if (!sec)
|
||||||
|
sec = 1;
|
||||||
|
else if (sec > 600 && cs->mask > UINT_MAX)
|
||||||
|
sec = 600;
|
||||||
|
|
||||||
|
clocks_calc_mult_shift(&cs->mult, &cs->shift, freq,
|
||||||
|
NSEC_PER_SEC / scale, sec * scale);
|
||||||
|
}
|
||||||
|
...
|
||||||
|
...
|
||||||
|
...
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Here we can see calculation of the maximum number of seconds which we can run before a clock source counter will overflow. First of all we fill the `sec` variable with the value of a clock source mask. Remember that a clock source's mask represents maximum amount of bits that are valid for the given clock source. After this, we can see two division operations. At first we divide our `sec` variable on a clock source frequency and than on scale factor. The `freq` parameter shows us how many timer interrupts will be occured in one second. So, we divide `mask` value that represents maximum number of a counter (for example `jiffy`) on the frequency of a timer and will get the maximum number of seconds for the certain clock source. The second division operation will give us maximum number of seconds for the certain clock source depends on its scale factor which can be `1` hertz or `1` kilohertz (10^ Hz).
|
||||||
|
|
||||||
|
After we have got maximum number of seconds, we check this value and set it to `1` or `600` depends on the result at the next step. These values is maximum sleeping time for a clocksource in seconds. In the next step we can see call of the `clocks_calc_mult_shift`. Main point of this function is calculation of the `mult` and `shift` values for a given clock source. In the end of the `__clocksource_update_freq_scale` function we check that just calculated `mult` value of a given clock source will not cause overflow after adjustment, update the `max_idle_ns` and `max_cycles` values of a given clock source with the maximum nanoseconds that can be converted to a clock source counter and print result to the kernel buffer:
|
||||||
|
|
||||||
|
```C
|
||||||
|
pr_info("%s: mask: 0x%llx max_cycles: 0x%llx, max_idle_ns: %lld ns\n",
|
||||||
|
cs->name, cs->mask, cs->max_cycles, cs->max_idle_ns);
|
||||||
|
```
|
||||||
|
|
||||||
|
that we can see in the [dmesg](https://en.wikipedia.org/wiki/Dmesg) output:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ dmesg | grep "clocksource:"
|
||||||
|
[ 0.000000] clocksource: refined-jiffies: mask: 0xffffffff max_cycles: 0xffffffff, max_idle_ns: 1910969940391419 ns
|
||||||
|
[ 0.000000] clocksource: hpet: mask: 0xffffffff max_cycles: 0xffffffff, max_idle_ns: 133484882848 ns
|
||||||
|
[ 0.094084] clocksource: jiffies: mask: 0xffffffff max_cycles: 0xffffffff, max_idle_ns: 1911260446275000 ns
|
||||||
|
[ 0.205302] clocksource: acpi_pm: mask: 0xffffff max_cycles: 0xffffff, max_idle_ns: 2085701024 ns
|
||||||
|
[ 1.452979] clocksource: tsc: mask: 0xffffffffffffffff max_cycles: 0x7350b459580, max_idle_ns: 881591204237 ns
|
||||||
|
```
|
||||||
|
|
||||||
|
After the `__clocksource_update_freq_scale` function will finish its work, we can return back to the `__clocksource_register_scale` function that will register new clock source. We can see the call of the following three functions:
|
||||||
|
|
||||||
|
```C
|
||||||
|
mutex_lock(&clocksource_mutex);
|
||||||
|
clocksource_enqueue(cs);
|
||||||
|
clocksource_enqueue_watchdog(cs);
|
||||||
|
clocksource_select();
|
||||||
|
mutex_unlock(&clocksource_mutex);
|
||||||
|
```
|
||||||
|
|
||||||
|
Note that before the first will be called, we lock the `clocksource_mutex` [mutex](https://en.wikipedia.org/wiki/Mutual_exclusion). The point of the `clocksource_mutex` mutex is to protect `curr_clocksource` variable which represents currently selected `clocksource` and `clocksource_list` variable which represents list that contains registered `clocksources`. Now, let's look on these three functions.
|
||||||
|
|
||||||
|
The first `clocksource_enqueue` function and other two defined in the same source code [file](https://github.com/torvalds/linux/tree/master/kernel/time/clocksource.c). We go through all already registered `clocksources` or in other words we go through all elements of the `clocksource_list` and tries to find best place for a given `clocksource`:
|
||||||
|
|
||||||
|
```C
|
||||||
|
static void clocksource_enqueue(struct clocksource *cs)
|
||||||
|
{
|
||||||
|
struct list_head *entry = &clocksource_list;
|
||||||
|
struct clocksource *tmp;
|
||||||
|
|
||||||
|
list_for_each_entry(tmp, &clocksource_list, list)
|
||||||
|
if (tmp->rating >= cs->rating)
|
||||||
|
entry = &tmp->list;
|
||||||
|
list_add(&cs->list, entry);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
In the end we just insert new clocksource to the `clocksource_list`. The second function - `clocksource_enqueue_watchdog` does almost the same that previous function, but it inserts new clock source to the `wd_list` depends on flangs of a clock source and starts new [watchdog](https://en.wikipedia.org/wiki/Watchdog_timer) timer. As I already wrote, we will not consider `watchdog` related stuff in this part but will do it in next parts.
|
||||||
|
|
||||||
|
The last function is the `clocksource_select`. As we can understand from the function's name, main point of this function - select the best `clocksource` from registered clocksources. This function consists only from the call of the function helper:
|
||||||
|
|
||||||
|
```C
|
||||||
|
static void clocksource_select(void)
|
||||||
|
{
|
||||||
|
return __clocksource_select(false);
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Note that the `__clocksource_select` function takes one parameter (`false` in our case). This [bool](https://en.wikipedia.org/wiki/Boolean_data_type) parameter shows how to traverese the `clocksource_list`. In our case we pass `false` that is mean that we will go through all entries of the `clocksource_list`. We already know that `clocksource` with the best rating will the first in the `clocksource_list` after the call of the `clocksource_enqueue` function, so we can easily get it from this list. After we found a clock source with the best rating, we switch to it:
|
||||||
|
|
||||||
|
```C
|
||||||
|
if (curr_clocksource != best && !timekeeping_notify(best)) {
|
||||||
|
pr_info("Switched to clocksource %s\n", best->name);
|
||||||
|
curr_clocksource = best;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
The result of this operation we can see in the `dmesg` output:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ dmesg | grep Switched
|
||||||
|
[ 0.199688] clocksource: Switched to clocksource hpet
|
||||||
|
[ 2.452966] clocksource: Switched to clocksource tsc
|
||||||
|
```
|
||||||
|
|
||||||
|
Note that we can see two clock sources in the `dmesg` output (`hpet` and `tsc` in our case). Yes, actually there can be many different clock sources on a particular hardware. So the Linux kernel knows about all registered clock sources and switches to a clock source with a better rating each time after registration of a new clock source.
|
||||||
|
|
||||||
|
If we will look on the bottom of the [kernel/time/clocksource.c](https://github.com/torvalds/linux/tree/master/kernel/time/clocksource.c) source code file, we will see that it has [sysfs](https://en.wikipedia.org/wiki/Sysfs) interface. Main initialization occurs in the `init_clocksource_sysfs` function which will be called during device `initcalls`. Let's look on the implementation of the `init_clocksource_sysfs` function:
|
||||||
|
|
||||||
|
```C
|
||||||
|
static struct bus_type clocksource_subsys = {
|
||||||
|
.name = "clocksource",
|
||||||
|
.dev_name = "clocksource",
|
||||||
|
};
|
||||||
|
|
||||||
|
static int __init init_clocksource_sysfs(void)
|
||||||
|
{
|
||||||
|
int error = subsys_system_register(&clocksource_subsys, NULL);
|
||||||
|
|
||||||
|
if (!error)
|
||||||
|
error = device_register(&device_clocksource);
|
||||||
|
if (!error)
|
||||||
|
error = device_create_file(
|
||||||
|
&device_clocksource,
|
||||||
|
&dev_attr_current_clocksource);
|
||||||
|
if (!error)
|
||||||
|
error = device_create_file(&device_clocksource,
|
||||||
|
&dev_attr_unbind_clocksource);
|
||||||
|
if (!error)
|
||||||
|
error = device_create_file(
|
||||||
|
&device_clocksource,
|
||||||
|
&dev_attr_available_clocksource);
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
device_initcall(init_clocksource_sysfs);
|
||||||
|
```
|
||||||
|
|
||||||
|
First of all we can see that it registers a `clocksource` subsystem with the call of the `subsys_system_register` function. In other words, after the call of this function, we will have following directory:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ pwd
|
||||||
|
/sys/devices/system/clocksource
|
||||||
|
```
|
||||||
|
|
||||||
|
After this step, we can see registration of the `device_clocksource` device which is represented by the following structure:
|
||||||
|
|
||||||
|
```C
|
||||||
|
static struct device device_clocksource = {
|
||||||
|
.id = 0,
|
||||||
|
.bus = &clocksource_subsys,
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
and creation of three files:
|
||||||
|
|
||||||
|
* `dev_attr_current_clocksource`;
|
||||||
|
* `dev_attr_unbind_clocksource`;
|
||||||
|
* `dev_attr_available_clocksource`.
|
||||||
|
|
||||||
|
These files will provide information about current clock source in the system, available clock sources in the system and interface which allows to unbind the clock source.
|
||||||
|
|
||||||
|
After the `init_clocksource_sysfs` function will be executed, we will be able find some information about avaliable clock sources in the:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ cat /sys/devices/system/clocksource/clocksource0/available_clocksource
|
||||||
|
tsc hpet acpi_pm
|
||||||
|
```
|
||||||
|
|
||||||
|
Or for example informantion about current clock source in the system:
|
||||||
|
|
||||||
|
```
|
||||||
|
$ cat /sys/devices/system/clocksource/clocksource0/current_clocksource
|
||||||
|
tsc
|
||||||
|
```
|
||||||
|
|
||||||
|
In the previous part, we saw API for the registration of the `jiffies` clock source, but didn't dive into details about the `clocksource` framework. In this part we did it and saw implementation of the new clock source registration and selection of a clock source with the best rating value in the system. Of course, this is not all API that `clocksource` framework provides. There a couple additional functions like `clocksource_unregister` for removing given clock source from the `clocksource_list` and etc. But I will not describe this functions in this part, because they are not important for us right now. Anyway if you are interesting in it, you can find it in the [kernel/time/clocksource.c](https://github.com/torvalds/linux/tree/master/kernel/time/clocksource.c).
|
||||||
|
|
||||||
|
That's all.
|
||||||
|
|
||||||
|
Conclusion
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
This is the end of the second part of the chapter that describes timers and timer management related stuff in the Linux kernel. In the previous part got acquainted with the following two concepts: `jiffies` and `clocksource`. In this part we saw some examples of the `jiffies` usage and knew more details about the `clocksource` concept.
|
||||||
|
|
||||||
|
If you have questions or suggestions, feel free to ping me in twitter [0xAX](https://twitter.com/0xAX), drop me [email](anotherworldofworld@gmail.com) or just create [issue](https://github.com/0xAX/linux-internals/issues/new).
|
||||||
|
|
||||||
|
**Please note that English is not my first language and I am really sorry for any inconvenience. If you found any mistakes please send me PR to [linux-insides](https://github.com/0xAX/linux-insides).**
|
||||||
|
|
||||||
|
Links
|
||||||
|
-------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
* [x86](https://en.wikipedia.org/wiki/X86)
|
||||||
|
* [x86_64](https://en.wikipedia.org/wiki/X86-64)
|
||||||
|
* [uptime](https://en.wikipedia.org/wiki/Uptime)
|
||||||
|
* [Ensoniq Soundscape Elite](https://en.wikipedia.org/wiki/Ensoniq_Soundscape_Elite)
|
||||||
|
* [RTC](https://en.wikipedia.org/wiki/Real-time_clock)
|
||||||
|
* [interrupts](https://en.wikipedia.org/wiki/Interrupt)
|
||||||
|
* [IBM PC](https://en.wikipedia.org/wiki/IBM_Personal_Computer)
|
||||||
|
* [programmable interval timer](https://en.wikipedia.org/wiki/Programmable_interval_timer)
|
||||||
|
* [Hz](https://en.wikipedia.org/wiki/Hertz)
|
||||||
|
* [nanoseconds](https://en.wikipedia.org/wiki/Nanosecond)
|
||||||
|
* [dmesg](https://en.wikipedia.org/wiki/Dmesg)
|
||||||
|
* [time stamp counter](https://en.wikipedia.org/wiki/Time_Stamp_Counter)
|
||||||
|
* [loadable kernel module](https://en.wikipedia.org/wiki/Loadable_kernel_module)
|
||||||
|
* [IA64](https://en.wikipedia.org/wiki/IA-64)
|
||||||
|
* [watchdog](https://en.wikipedia.org/wiki/Watchdog_timer)
|
||||||
|
* [clock rate](https://en.wikipedia.org/wiki/Clock_rate)
|
||||||
|
* [mutex](https://en.wikipedia.org/wiki/Mutual_exclusion)
|
||||||
|
* [sysfs](https://en.wikipedia.org/wiki/Sysfs)
|
||||||
|
* [previous part](https://0xax.gitbooks.io/linux-insides/content/Timers/timers-1.html)
|
@ -74,3 +74,4 @@ Thank you to all contributors:
|
|||||||
* [Dennis Birkholz](https://github.com/dennisbirkholz)
|
* [Dennis Birkholz](https://github.com/dennisbirkholz)
|
||||||
* [Anton Tyurin](https://github.com/noxiouz)
|
* [Anton Tyurin](https://github.com/noxiouz)
|
||||||
* [Bogdan Kulbida](https://github.com/kulbida)
|
* [Bogdan Kulbida](https://github.com/kulbida)
|
||||||
|
* [Matt Hudgins](https://github.com/mhudgins)
|
||||||
|
Loading…
Reference in New Issue
Block a user