mirror of
https://github.com/0xAX/linux-insides.git
synced 2025-01-08 23:01:05 +00:00
mark some constants
This commit is contained in:
parent
896411ad17
commit
af4d2701fa
@ -34,7 +34,7 @@ CS selector 0xf000
|
|||||||
CS base 0xffff0000
|
CS base 0xffff0000
|
||||||
```
|
```
|
||||||
|
|
||||||
The processor starts working in [real mode](https://en.wikipedia.org/wiki/Real_mode). Let's back up a little and try to understand memory segmentation in this mode. Real mode is supported on all x86-compatible processors, from the [8086](https://en.wikipedia.org/wiki/Intel_8086) all the way to the modern Intel 64-bit CPUs. The 8086 processor has a 20-bit address bus, which means that it could work with a 0-0xFFFFF address space (1 megabyte). But it only has 16-bit registers, which have a maximum address of 2^16 - 1 or 0xffff (64 kilobytes). [Memory segmentation](http://en.wikipedia.org/wiki/Memory_segmentation) is used to make use of all the address space available. All memory is divided into small, fixed-size segments of 65536 bytes (64 KB). Since we cannot address memory above 64 KB with 16 bit registers, an alternate method is devised. An address consists of two parts: a segment selector, which has a base address, and an offset from this base address. In real mode, the associated base address of a segment selector is `Segment Selector * 16`. Thus, to get a physical address in memory, we need to multiply the segment selector part by 16 and add the offset:
|
The processor starts working in [real mode](https://en.wikipedia.org/wiki/Real_mode). Let's back up a little and try to understand memory segmentation in this mode. Real mode is supported on all x86-compatible processors, from the [8086](https://en.wikipedia.org/wiki/Intel_8086) all the way to the modern Intel 64-bit CPUs. The 8086 processor has a 20-bit address bus, which means that it could work with a 0-0xFFFFF address space (1 megabyte). But it only has 16-bit registers, which have a maximum address of `2^16 - 1` or `0xffff` (64 kilobytes). [Memory segmentation](http://en.wikipedia.org/wiki/Memory_segmentation) is used to make use of all the address space available. All memory is divided into small, fixed-size segments of 65536 bytes (64 KB). Since we cannot address memory above 64 KB with 16 bit registers, an alternate method is devised. An address consists of two parts: a segment selector, which has a base address, and an offset from this base address. In real mode, the associated base address of a segment selector is `Segment Selector * 16`. Thus, to get a physical address in memory, we need to multiply the segment selector part by 16 and add the offset:
|
||||||
|
|
||||||
```
|
```
|
||||||
PhysicalAddress = Segment Selector * 16 + Offset
|
PhysicalAddress = Segment Selector * 16 + Offset
|
||||||
@ -58,7 +58,7 @@ which is 65520 bytes past the first megabyte. Since only one megabyte is accessi
|
|||||||
|
|
||||||
Ok, now we know about real mode and memory addressing. Let's get back to discussing register values after reset:
|
Ok, now we know about real mode and memory addressing. Let's get back to discussing register values after reset:
|
||||||
|
|
||||||
The `CS` register consists of two parts: the visible segment selector, and the hidden base address. While the base address is normally formed by multiplying the segment selector value by 16, during a hardware reset the segment selector in the CS register is loaded with 0xf000 and the base address is loaded with 0xffff0000; the processor uses this special base address until `CS` is changed.
|
The `CS` register consists of two parts: the visible segment selector, and the hidden base address. While the base address is normally formed by multiplying the segment selector value by 16, during a hardware reset the segment selector in the CS register is loaded with `0xf000` and the base address is loaded with `0xffff0000`; the processor uses this special base address until `CS` is changed.
|
||||||
|
|
||||||
The starting address is formed by adding the base address to the value in the EIP register:
|
The starting address is formed by adding the base address to the value in the EIP register:
|
||||||
|
|
||||||
@ -79,7 +79,7 @@ reset_vector:
|
|||||||
...
|
...
|
||||||
```
|
```
|
||||||
|
|
||||||
Here we can see the `jmp` instruction [opcode](http://ref.x86asm.net/coder32.html#xE9), which is 0xe9, and its destination address at `_start - ( . + 2)`. We can also see that the `reset` section is 16 bytes, and that it starts at `0xfffffff0`:
|
Here we can see the `jmp` instruction [opcode](http://ref.x86asm.net/coder32.html#xE9), which is `0xe9`, and its destination address at `_start - ( . + 2)`. We can also see that the `reset` section is 16 bytes, and that it starts at `0xfffffff0`:
|
||||||
|
|
||||||
```
|
```
|
||||||
SECTIONS {
|
SECTIONS {
|
||||||
@ -379,13 +379,13 @@ Almost all of the setup code is in preparation for the C language environment in
|
|||||||
|
|
||||||
This can lead to 3 different scenarios:
|
This can lead to 3 different scenarios:
|
||||||
|
|
||||||
* `ss` has valid value 0x10000 (as do all other segment registers beside `cs`)
|
* `ss` has valid value `0x10000` (as do all other segment registers beside `cs`)
|
||||||
* `ss` is invalid and `CAN_USE_HEAP` flag is set (see below)
|
* `ss` is invalid and `CAN_USE_HEAP` flag is set (see below)
|
||||||
* `ss` is invalid and `CAN_USE_HEAP` flag is not set (see below)
|
* `ss` is invalid and `CAN_USE_HEAP` flag is not set (see below)
|
||||||
|
|
||||||
Let's look at all three of these scenarios in turn:
|
Let's look at all three of these scenarios in turn:
|
||||||
|
|
||||||
* `ss` has a correct address (0x10000). In this case, we go to label [2](https://github.com/torvalds/linux/blob/master/arch/x86/boot/header.S#L481):
|
* `ss` has a correct address (`0x10000`). In this case, we go to label [2](https://github.com/torvalds/linux/blob/master/arch/x86/boot/header.S#L481):
|
||||||
|
|
||||||
```assembly
|
```assembly
|
||||||
2: andw $~3, %dx
|
2: andw $~3, %dx
|
||||||
|
Loading…
Reference in New Issue
Block a user