# HE<sup>2</sup>B MICL: TD02: Instructions logiques et de manipulation de bits

Année académique 2020 – 2021

Ce deuxième TD commence par la description du registre rflags. La découverte des instructions logiques not, and, or et xor suit. Les instructions de manipulation de bits bt, btr, bts et btc sont finalement étudiées.

## 1 Valeurs booléennes

Les valeurs booléennes <sup>1</sup> sont au nombre de deux : *Vrai* et *Faux*. Un seul bit suffit donc pour représenter une variable booléenne. Par convention, la valeur *Vrai* est codée par un bit à 1 tandis que *Faux* est représenté par un bit à 0.

# 2 Registre rflags

Le registre rflags<sup>2</sup> est un registre de 64 bits dont certains sont des indicateurs (drapeaux, flags). La Fig. 1 illustre le contenu de eflags, le registre des états des processeurs x86 32 bits. rflags en constitue une extension sur 64 bits. Les 32 nouveaux bits sont de poids 32 à 63. Ils sont tous réservés<sup>3</sup>.

 $<sup>^*\</sup>mathrm{Et}$ aussi, lors des années passées : ABS – BEJ – DWI – EGR – ELV – FPL – JDS – MBA – MCD – MHI – MWA.

<sup>1.</sup> https://fr.wikipedia.org/wiki/Bool%C3%A9en (consulté le 30 janvier 2020).

<sup>2.</sup> https://fr.wikipedia.org/wiki/RFLAGS (consulté le 30 janvier 2020).

<sup>3.</sup> Voir le manuel d'Intel [1, section 3.4.3.4, p. 3-18 Vol. 1] : In 64-bit mode, EFLAGS is extended to 64 bits and called RFLAGS. The upper 32 bits of RFLAGS register is reserved. The lower 32 bits of RFLAGS is the same as EFLAGS.



FIG. 1 – Le registre eflags (Illustration © Intel: Intel® 64 and IA-32 Architectures Software Developer's Manual, Combined Volumes: 1, 2A, 2B, 2C, 2D, 3A, 3B, 3C, 3D and 4<sup>a</sup>, p. 3-16 Vol. 1).

```
a. https://software.intel.com/sites/default/files/managed/39/c5/
325462-sdm-vol-1-2abcd-3abcd.pdf (consulté le 30 janvier 2020).
```

Les indicateurs fournissent des informations sur le déroulement du processus en cours, notamment sur le résultat de certaines opérations. Ils sont appelés « indicateurs d'état » ( $status\ flags$ ) car ils permettent notamment de savoir comment se sont déroulées les opérations arithmétiques  $^4$ .

Nous ne voyons dans ce laboratoire que quatre indicateurs :

- Carry flag (CF): indicateur de retenue<sup>5</sup>, bit de rang 0 de rflags;
- Zero flag (ZF) : indicateur de zéro <sup>6</sup>, bit de rang 6 de rflags;
- Sign flag (SF): indicateur de signe<sup>7</sup>, bit de rang 7 de rflags;
- Overflow flag (OF): indicateur de débordement<sup>8</sup>, bit de rang 11 de rflags.

L'accès direct au contenu de **rflags** n'est pas possible. On accède indirectement à son contenu via des instructions particulières, telles celles de saut (TD03 et TD06). Pour cette raison, il n'est pas nécessaire de connaître la position précise d'un *flaq* dans le

<sup>4.</sup> Les indicateurs d'état sont au nombre de six, mais seuls quatre d'entre eux sont vus dans ce labo. Outre les indicateurs d'état, il existe d'autres indicateurs, non vus lors des labos micro : les « indicateurs de contrôle » (control flags), associés aux instructions de manipulation de chaînes de caractères, et les « indicateurs système » (system flags), utilisés par le système d'exploitation.

<sup>5.</sup> https://fr.wikipedia.org/wiki/Indicateur\_de\_retenue (consulté le 30 janvier 2020).

<sup>6.</sup> https://en.wikipedia.org/wiki/Zero\_flag (consulté le 30 janvier 2020).

<sup>7.</sup> https://en.wikipedia.org/wiki/Sign\_flag (consulté le 30 janvier 2020).

<sup>8.</sup> https://fr.wikipedia.org/wiki/Indicateur\_de\_d%C3%A9bordement (consulté le 30 janvier 2020).

| Instruction | Effet                         | Contraintes                                            | Flags affectés |  |
|-------------|-------------------------------|--------------------------------------------------------|----------------|--|
| not X       | Inverse tous les<br>bits de X | X = registre ou<br>variable de 8, 16,<br>32 ou 64 bits | Aucun          |  |

(a) Résumé.

(b) Exemple.

Table 1 – Instruction not.

registre rflags.

# 3 Instructions logiques

#### **3.1** not

L'instruction not 9 n'a qu'un opérande qui joue le rôle de source et de destination et qui doit être un registre ou une variable (voir TD04 et TD07) de 8, 16, 32 ou 64 bits. not a pour effet d'inverser  $^{10}$  tous les bits de son opérande. Cette opération est également appelée complément à  $1^{11}$ .

La Table 1(a) donne un résumé de cette instruction, et la Table 1(b) en donne un exemple.

### 3.2 and, or et xor

Les instructions and <sup>12</sup>, or <sup>13</sup> et xor <sup>14</sup> ont deux opérandes : la *destination*, à gauche de la virgule, et la *source*, à droite. Ils peuvent être des registres ou des variables de 8, 16, 32 ou 64 bits, mais pas tous les deux des emplacements mémoire. En outre, la source peut être un immédiat.

Ces instructions effectuent, respectivement, un  $et^{15}$ , un  $ou^{16}$  et un ou exclusif<sup>17</sup> logiques bit à bit entre la source et la destination. Ils placent le résultat dans la destination,

```
9. https://www.felixcloutier.com/x86/not (consulté le 30 janvier 2020).
```

<sup>10.</sup> https://fr.wikipedia.org/wiki/Fonction\_NON (consulté le 30 janvier 2020).

<sup>11.</sup> https://fr.wikipedia.org/wiki/Compl%C3%A9ment\_%C3%A0\_un (consulté le 30 janvier 2020).

<sup>12.</sup> https://www.felixcloutier.com/x86/and (consulté le 30 janvier 2020).

<sup>13.</sup> https://www.felixcloutier.com/x86/or (consulté le 30 janvier 2020).

<sup>14.</sup> https://www.felixcloutier.com/x86/xor (consulté le 30 janvier 2020).

<sup>15.</sup> https://fr.wikipedia.org/wiki/Fonction\_ET (consulté le 30 janvier 2020).

<sup>16.</sup> https://fr.wikipedia.org/wiki/Fonction\_OU (consulté le 30 janvier 2020).

<sup>17.</sup> https://fr.wikipedia.org/wiki/Fonction\_OU\_exclusif (consulté le 30 janvier 2020).

| Instruction | Effet                                                                                                          | Contraintes                                     | Flags affectés                                                                                                                       |
|-------------|----------------------------------------------------------------------------------------------------------------|-------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------|
| and X, Y    | $\begin{array}{c} \textbf{X} \leftarrow \textbf{X} \ \text{ET} \ \textbf{Y} \\ \text{(bit à bit)} \end{array}$ | Registres ou variables de 8, 16, 32 ou 64 bits, | $CF \leftarrow 0$                                                                                                                    |
| or X, Y     | $\begin{array}{c} X \leftarrow X \text{ OU Y} \\ \text{(bit à bit)} \end{array}$                               | pas deux variables, Y peut être un immédiat     | $\begin{array}{c} \texttt{OF} \leftarrow 0 \\ \texttt{SF} \leftarrow \texttt{bit de signe}^{a} \ \texttt{du r\'esultat} \end{array}$ |
| xor X, Y    | $X \leftarrow X \text{ XOR } Y$ (bit à bit)                                                                    | (8, 16 ou 32 bits)                              | $ZF \leftarrow 1$ si résultat nul, 0 sinon                                                                                           |

<sup>(</sup>a) Résumé.

a. Le bit de signe d'un motif binaire est son bit de rang le plus élevé.

| al | : 11100101 <i>b</i>       |                                                                                                              |
|----|---------------------------|--------------------------------------------------------------------------------------------------------------|
| ah | : $10101010b$             |                                                                                                              |
| ah |                           |                                                                                                              |
| al | : 10100000b               |                                                                                                              |
| al | : $11100101b$             |                                                                                                              |
| ah | : $01010101b$             |                                                                                                              |
| ah |                           |                                                                                                              |
| al | : 11110101 <i>b</i>       |                                                                                                              |
| :  | 1100010011100101 <i>b</i> |                                                                                                              |
| :  | 0011000001100010b         |                                                                                                              |
|    |                           |                                                                                                              |
| :  | 1111010010000111b         |                                                                                                              |
|    | ah ah al ah ah ah ah :    | al : 10100000b  al : 11100101b  ah : 01010101b  ah  al : 11110101b  : 1100010011100101b  : 0011000001100010b |

(b) Exemples.

TABLE 2 – Instructions and, or et xor.

sans modifier la source.

En plus, elles modifient le  $sign\ flag\ (SF)$ , qui reçoit le bit de rang le plus élevé du résultat  $^{18}$ , et le  $zero\ flag\ (ZF)$ , qui indique si le résultat est nul : 1 pour nul, 0 pour non nul. Le  $carry\ flag\ (CF)$  et le  $overflow\ flag\ (OF)$  sont mis à 0.

La Table 2(a) donne un résumé de ces instructions, et la Table 2(b) en donne des exemples d'utilisation.

**Remarque** Comme indiqué dans la TABLE 2(a), les immédiats s'étendent sur maximum 32 bits. Si on utilise un immédiat avec une destination dont la taille fait 64 bits, l'immédiat est étendu sur 64 bits par extension de signe <sup>19</sup>: son bit de signe, celui de rang 31, est recopié en bits 32 à 63.

Le code source suivant illustre ceci:

<sup>19.</sup> https://fr.wiktionary.org/wiki/extension\_de\_signe (consulté le 30 janvier 2020).



<sup>18.</sup> En représentation complément à deux, le bit de rang le plus élevé est un bit de signe : il vaut 0 pour un nombre positif, 1 pour un nombre négatif, d'où le nom de  $sign\ flag$ .

```
; O_extension_signe.asm
2
   global main
4
   section .text
   main:
       mov rax, 0xF0_F0_F0_F0_F0_F0_F0
       and rax, 0x80_00_00 ; attention : extension de signe
               ; rax : 0xF0_F0_F0_F0_80_00_00
10
       mov rsi, 0xF0_F0_F0_F0_F0_F0_F0
11
       mov rdi, 0xFF_FF_FF_FF_80_00_00_00
12
       and rsi, rdi
                               ; idem
13
               ; rsi : 0xF0_F0_F0_F0_80_00_00
14
15
       mov rbx, 0xF0_F0_F0_F0_F0_F0_F0
16
       mov rcx, 0x80_00_00_00
17
       and rbx, rcx
18
               ; rbx : 0x00_00_00_00_80_00_00
20
       ; fin
21
       mov rax, 60
22
       mov rdi, 0
23
       syscall
```

Dès lors, si les deux opérandes font 64 bits, il est impossible d'utiliser un immédiat comme source, mais il faut utiliser un registre ou une variable.

## 3.3 Masquage

Le masquage <sup>20</sup> consiste à effectuer une opération logique afin de conserver certains bits d'un opérande, et d'en modifier d'autres. Nous effectuons des masquages à l'aide des instructions and, or et xor.

#### 3.3.1 Masque avec and

L'instruction and permet de conserver certains bits d'un opérande et de mettre les autres à  $0^{21}$ .

Pour ce faire, on fait un and entre cet opérande et un second opérande appelé masque. Ce masque est constitué de bits à 1 aux positions qu'on désire conserver et de bits à 0

<sup>20.</sup> https://en.wikipedia.org/wiki/Mask\_(computing) (consulté le 30 janvier 2020).

<sup>21.</sup> https://en.wikipedia.org/wiki/Mask\_(computing)#Masking\_bits\_to\_0 (consulté le 30 janvier 2020).

aux positions qu'on veut mettre à 0.

Par exemple, si on désire conserver les quatre bits de droite de al et mettre les autres à 0, on utilise le masque 00001111b:

al : 11100101bbl (masque) : 00001111band al, bl al : 00000101b

#### 3.3.2 Masque avec or

L'instruction or permet de conserver certains bits d'un opérande et de mettre les autres à  $1^{22}$ .

Pour ce faire, on fait un or entre cet opérande et le masque. Ce masque est constitué de bits à 0 aux positions qu'on désire conserver et de bits à 1 aux positions qu'on veut mettre à 1.

Par exemple, si on désire conserver les quatre bits de droite de **al** et mettre les autres à 1, on utilise le masque 11110000b:

al : 11100101bbl (masque) : 11110000bor al, bl al : 11110101b

#### 3.3.3 Masque avec xor

L'instruction **xor** permet de conserver certains bits d'un opérande et d'inverser les autres <sup>23</sup>.

Pour ce faire, on fait un **xor** entre cet opérande et le masque. Ce masque est constitué de bits à 0 aux positions qu'on désire conserver et de bits à 1 aux positions qu'on veut inverser.

Par exemple, si on désire conserver les quatre bits de droite de al et inverser les autres, on utilise le masque 11110000b:

al : 11100101bbl (masque) : 11110000bxor al, bl al : 00010101b



<sup>22.</sup> https://en.wikipedia.org/wiki/Mask\_(computing)#Masking\_bits\_to\_1 (consulté le 30 janvier 2020).

<sup>23.</sup> https://en.wikipedia.org/wiki/Mask\_(computing)#Toggling\_bit\_values (consulté le 30 janvier 2020).

| Instruction | Effet               | Contraintes             | Flags affectés                  |  |  |
|-------------|---------------------|-------------------------|---------------------------------|--|--|
| bt X, Y     | Aucun               | X = registre ou         | $CF \leftarrow valeur initiale$ |  |  |
| bts X, Y    | Le bit de rang Y de | variable de 16, 32 ou   | du bit de rang Y de X           |  |  |
| 505 A, 1    | X est mis à 1       | 64 bits                 |                                 |  |  |
| btr X, Y    | Le bit de rang Y de | Y = registre de  16, 32 | ZF n'est pas modifié            |  |  |
|             | X est mis à 0       | ou 64 bits ou           | _                               |  |  |
| btc X, Y    | Le bit de rang Y de | immédiat sur 8 bits     | OF et SF sont indéfinis         |  |  |
|             | X est complémenté   |                         |                                 |  |  |

(a) Résumé.



(b) Exemples.

Table 3 – Instructions bt, bts, btr et btc.

## 4 Instructions de manipulation de bits : bt, bts, btr, btc

L'instruction bt <sup>24</sup> (bit test) teste un bit précis d'un motif binaire donné. Les instructions bts <sup>25</sup> (bit test and set), btr <sup>26</sup> (bit test and reset) et btc <sup>27</sup> (bit test and complement) testent également un bit avant de le mettre à 1, 0 ou de le complémenter, respectivement.

Le premier opérande de ces instructions est un registre ou une variable de 16, 32 ou 64 bits. Leur second opérande est un registre de 16, 32 ou 64 bits ou un immédiat sur 8



<sup>24.</sup> https://www.felixcloutier.com/x86/bt (consulté le 30 janvier 2020).

<sup>25.</sup> https://www.felixcloutier.com/x86/bts (consulté le 30 janvier 2020).

<sup>26.</sup> https://www.felixcloutier.com/x86/btr (consulté le 30 janvier 2020).

<sup>27.</sup> https://www.felixcloutier.com/x86/btc (consulté le 30 janvier 2020).

bits. Si le deuxième opérande est un registre, il doit être de même taille que le premier. Ces quatre instructions copient dans le *carry flag* (CF) le bit du premier opérande dont le rang est fourni via le second opérande. C'est la partie *test*, à laquelle se limite

bt. Ensuite, l'instruction :

- bts met ce bit du premier opérande à 1;
- btr met ce bit du premier opérande à 0;
- btc complémente ce bit du premier opérande.

Le zero flag (ZF) n'est pas modifié. L'overflow flag (OF) et le sign flag (SF) sont indéfinis. La Table 3(a) donne un résumé de ces instructions. La Table 3(b) en donne des exemples d'utilisation.

## 5 Exercices

Pour réaliser les exercices qui suivent, vous ne pouvez utiliser que les instructions étudiées au long des TD01 et 02 : mov, not, and, or, xor, bt, bts, btr, btc, syscall et nop.

**Ex. 1** Remplissez, à la main, c'est-à-dire sans employer l'ordinateur, les valeurs des registres et flags dans le code ci-dessous. Dans un deuxième temps, exécutez ce programme dans KDbg (le code source est fourni en annexe). Vérifiez que le contenu des registres, y compris rflags, est conforme à vos réponses.

```
; 1_comprehension_log.asm
   global main
3
   section .text
   main:
        mov al, 10011101b
                                ; al = \ldots, zf = \ldots, sf = \ldots
        not al
9
        mov al, 11100101b
10
        mov ah, 00101010b
11
                               ; al = \ldots, zf = \ldots, sf = \ldots
        and al, ah
12
13
        mov al, 11100101b
14
        mov ah, 00001010b
15
        and al, ah
                               ; al = \ldots, zf = \ldots, sf = \ldots
16
17
        mov al, 01100101b
        mov ah, 01010101b
19
                               ; al = \ldots, zf = \ldots, sf = \ldots
20
            al, ah
21
```

```
mov al, 11100101b
22
        mov ah, 01010101b
23
                               ; al = \ldots, zf = \ldots, sf = \ldots
            al, ah
24
25
        mov dx, 1100010011100101b
26
        mov si, 0011000001100010b
27
        xor dx, si
                               ; dx = \dots \dots
28
                               ; zf = ., sf = .
29
30
        mov al, 11100101b
31
        mov ah, 11100101b
32
                               ; al = \ldots, zf = \ldots, sf = \ldots
        xor al, ah
33
34
        ; fin
35
        mov rax, 60
36
        mov rdi, 0
37
        syscall
```

**Notes sur KDbg** Le contenu de **rflags** apparaît dans la section *Flags* de la vue (*View*) *Registers*. Il est possible de basculer entre une vue binaire et une vue plus explicite nommant les drapeaux levés en cliquant droit sur la ligne **eflags** et en sélectionnant *Binaire* ou *Défaut GDB*, respectivement, dans le menu surgissant.

Il est possible d'observer le contenu des registres comme rax dans la vue *Registers*. Cependant, si on n'est intéressé que par le contenu de ah et qu'on n'a pas envie de fouiller dans la représentation binaire de rax, il est possible de *surveiller* ah. Pour ce faire, il faut ouvrir la vue *Expressions surveillées* (*Watched Expressions*). Dans la zone d'édition, fournissez le nom du registre à guetter précédé du symbole \$ <sup>28</sup>.

Attention cependant, les registres r8b, r9b, etc. jusque r15b sont inaccessibles via ces noms dans KDbg. Il faut utiliser les noms r81, r91, etc. jusque r151 pour accéder à leurs contenus. Donc, pour voir le contenu de r8, r8d, r8w et r8b dans KDbg, il faut surveiller les expressions \$r8, \$r8d, \$r8w et \$r81, respectivement.

Il est par ailleurs possible de choisir le format d'affichage <sup>29</sup> des informations. Ainsi, pour examiner le contenu de ah en binaire, on doit fournir l'expression : /t \$ah tandis que pour voir le contenu de ce même registre comme un caractère, il faut surveiller l'expression : /c \$ah.



<sup>28.</sup> https://sourceware.org/gdb/current/onlinedocs/gdb/Registers.html#Registers (consulté le 30 janvier 2020).

<sup>29.</sup> https://sourceware.org/gdb/current/onlinedocs/gdb/Output-Formats.html#Output-Formats (consulté le 30 janvier 2020).

| b <sub>7</sub> ——— |         |         |                         | -                       | <b>→</b>             | 0 0 | 0 0 | 0 1 | 0 1 | 1 0 | 1 0 | 1 1 | 1 1 |
|--------------------|---------|---------|-------------------------|-------------------------|----------------------|-----|-----|-----|-----|-----|-----|-----|-----|
| b <sub>5</sub>     | _       |         |                         |                         | · •                  | 0   | 1   | 0   | 1   | 0   | 1   | 0   | 1   |
| Bits               | b₄<br>↓ | b₃<br>↓ | $_{\downarrow}^{b_{2}}$ | $_{\downarrow}^{b_{1}}$ | Column<br>→<br>Row ↓ | 0   | 1   | 2   | 3   | 4   | 5   | 6   | 7   |
|                    | 0       | 0       | 0                       | 0                       | 0                    | NUL | DLE | SP  | 0   | @   | P   | •   | р   |
|                    | 0       | 0       | 0                       | 1                       | 1                    | SOH | DC1 | ļ   | 1   | Α   | Q   | a   | q   |
|                    | 0       | 0       | 1                       | 0                       | 2                    | STX | DC2 |     | 2   | В   | R   | b   | r   |
|                    | 0       | 0       | 1                       | 1                       | 3                    | ETX | DC3 | #   | 3   | С   | S   | С   | S   |
|                    | 0       | 1       | 0                       | 0                       | 4                    | EOT | DC4 | \$  | 4   | D   | T   | d   | t   |
|                    | 0       | 1       | 0                       | 1                       | 5                    | ENQ | NAK | %   | 5   | E   | U   | е   | u   |
|                    | 0       | 1       | 1                       | 0                       | 6                    | ACK | SYN | &   | 6   | F   | V   | f   | V   |
|                    | 0       | 1       | 1                       | 1                       | 7                    | BEL | ETB | •   | 7   | G   | W   | g   | W   |
|                    | 1       | 0       | 0                       | 0                       | 8                    | BS  | CAN | (   | 8   | Н   | X   | h   | X   |
|                    | 1       | 0       | 0                       | 1                       | 9                    | HT  | EM  | )   | 9   | - 1 | Y   | į   | У   |
|                    | 1       | 0       | 1                       | 0                       | 10                   | LF  | SUB | *   | :   | J   | Z   | j   | Z   |
|                    | 1       | 0       | 1                       | 1                       | 11                   | VT  | ESC | +   |     | K   | [   | k   | {   |
|                    | 1       | 1       | 0                       | 0                       | 12                   | FF  | FC  | ,   | <   | L   | \   | - 1 |     |
|                    | 1       | 1       | 0                       | 1                       | 13                   | CR  | GS  | -   | =   | М   | ]   | m   | }   |
|                    | 1       | 1       | 1                       | 0                       | 14                   | SO  | RS  |     | >   | N   | ۸   | n   | ~   |
|                    | 1       | 1       | 1                       | 1                       | 15                   | SI  | US  | /   | ?   | 0   | _   | 0   | DEL |

Fig. 2 – Table ASCII (Illustration Wikipedia <sup>a</sup>).

**Ex. 2** En utilisant la table ASCII <sup>30</sup> présentée à la Fig. 2, remplissez les pointillés du code source suivant de manière à convertir le caractère minuscule en majuscule, à l'aide d'un masque (voir commentaires dans le code source).

```
; 2_ascii_minVersMaj_partiel.asm

global main

section .text
main:
nop ; ne fait rien

mov al, 'd'; on charge une lettre minuscule dans al

; ......; à compléter de sorte que ah contienne
; ......; la même lettre que al mais en majuscule.
; cela doit fonctionner sans modifier al,
; pour toutes les lettres. on suppose que
```

a. https://en.wikipedia.org/wiki/File:ASCII\_Code\_Chart-Quick\_ref\_card.png (consulté le 30 janvier 2020).

<sup>30.</sup> Attention!, la numérotation des bits qui y est utilisée commence à  $b_1$  au lieu de  $b_0$ , contrairement à la convention standard actuelle.

```
; le contenu de al est bien une lettre minuscule.

; fin
mov rax, 60
mov rdi, 0
syscall
```

**Ex. 3** Recodez le programme de l'exercice précédent à l'aide d'une ou plusieurs instructions de manipulation de bits au lieu d'utiliser des masques.

Ca ne compile pas? Utilisez bx comme destination plutôt que ah.

**Ex. 4** Écrivez un code qui, partant du contenu de **b1** dont on garantit qu'il s'agit d'un entier dans l'intervalle [0, 9], stocke dans **bh** le code ASCII du caractère représentant ce chiffre décimal.

## Notions à retenir

Registre rflags, indicateurs d'état CF, OF, SF et ZF, instructions logiques not, and, or et xor, masquage, instructions de manipulation de bits bt, bts, btr et btc.

# Références

- [1] Intel® 64 and IA-32 Architectures Software Developer's Manual, Combined Volumes: 1, 2A, 2B, 2C, 2D, 3A, 3B, 3C, 3D and 4, octobre 2017. https://software.intel.com/sites/default/files/managed/39/c5/325462-sdm-vol-1-2abcd-3abcd.pdf.
- [2] Igor Zhirkov. Low-Level Programming. Apress, 2017. https://www.apress.com/gp/book/9781484224021.