# Organización del Computador II

Departamento de Computación Facultad de Ciencias Exactas y Naturales Universidad de Buenos Aires

## Trabajo Práctico III

System Programming

## Grupo: Tú me pixeleas

| Integrante                 | LU     | Correo electrónico    |
|----------------------------|--------|-----------------------|
| Costa, Manuel José Joaquín | 035/14 | manuc94@hotmail.com   |
| Gatti, Mathias Nicolás     | 477/14 | mathigatti@gmail.com  |
| Pondal, Iván               | 078/14 | ivan.pondal@gmail.com |

# Índice

| 1.         | Intr | oducci  | ión                                  | 3  |
|------------|------|---------|--------------------------------------|----|
| <b>2</b> . | Des  | arrollo |                                      | 4  |
|            | 2.1. | Pasaje  | e a modo protegido y segmentación    | 4  |
|            |      | 2.1.1.  | Inicialización de la GDT             |    |
|            |      | 2.1.2.  | Pasaje a modo protegido              | 4  |
|            | 2.2. | Excep   | ciones                               | 4  |
|            |      | 2.2.1.  | Inicialización de la IDT             | ŀ  |
|            |      | 2.2.2.  | Carga de la IDT                      | ŀ  |
|            | 2.3. | Pagina  | ación básica                         | ŀ  |
|            |      | 2.3.1.  | Rutinas de mapeado y desmapeado      |    |
|            |      | 2.3.2.  | Generación del directorio de páginas | 6  |
|            |      | 2.3.3.  | Activación de la paginación          |    |
|            | 2.4. |         | ipciones externas                    |    |
|            |      | 2.4.1.  | Reloj                                |    |
|            |      | 2.4.2.  |                                      | 8  |
|            |      | 2.4.3.  | Servicios del sistema                | 8  |
|            | 2.5. | Tareas  | 8                                    | 8  |
|            |      | 2.5.1.  | Entradas de la GDT                   | 8  |
|            |      | 2.5.2.  | Inicialización de las TSS            | 8  |
|            |      | 2.5.3.  | Construcción de Tareas               | 8  |
|            |      | 2.5.4.  | La Interrupción 0x46                 | 8  |
| 3.         | Res  | ultado  | ${f s}$                              | 9  |
| 4.         | Con  | clusio  | nes 1                                | ın |

## 1. Introducción

El trabajo presentado consiste en la elaboración de un sistema mínimo mediante el cual se profundizaron los conceptos estudiados de *System Programming*. Para esto se siguieron una serie de ejercicios por los cuales en forma gradual se fue construyendo el producto final.

De forma análoga a la resolución de los ejercicios, la estructura del informe seguirá la paso por paso el desarrollo describiendo cada sección junto a la explicación de las decisiones que consideramos más importantes para el correcto funcionamiento del trabajo. A grandes rasgos, contaremos con un pasaje a modo protegido, configuración de segmentación e interrupciones, paginación, manejo de tareas y un scheduler.

El sistema resultante es un juego dentro del cual los jugadores deben soltar perros que buscan huesos y los llevan a su cucha, otorgándole así puntos al jugador correspondiente. El jugador con más huesos será el vencedor.

Para el desarrollo del mismo contamos con *Bochs* que nos permitió emular la computadora que cargaba y ejecutaba nuestro sistema, y en lo que respecta lenguajes de programación se utilizaron C y ASM alternándolos según la rutina que estuviéramos desarrollando.

## 2. Desarrollo

#### 2.1. Pasaje a modo protegido y segmentación

En esta sección explicaremos como realizamos la creación de la *Global Descriptor Table* (GDT) y como, una vez hecho esto, pasamos a modo protegido.

## 2.1.1. Inicialización de la GDT

Del código que viene dado se deben agregar en principio cinco segmentos (Para la parte de tareas tuvimos que agregar algunos mas los cuales serán explicados en su correspondiente sección), estos estan conformados por dos de datos, dos de código y uno de la pantalla de video, con niveles de privilegio de 0 o 3.

Para esto seteamos cada segmento, a partir del indice 8 de la gdt por restricciones del trabajo práctico. Las posiciones anteriores a la 8 quedan en 0. Los cuatro primeros segmentos direccionan los primeros 500 MB de memoria, desde la posición 0 hasta el megabyte número 500, en ese espacio entran 128.000 (0x1F400) bloques de 4 KB por lo tanto el limite sera 0x1F3FF ya que contamos desde el 0, los de codigo son del tipo 10 (0x0A) y los de datos del tipo (0x01), el limite es (0xF3FF) y el nivel de privilegio 0 o 3 segun corresponda. La granularidad esta activada (0x01). Los bits P, L, D/B y AVL se ponen en 1, 0, 1 y 0 respectivamente para todas las entradas.

#### 2.1.2. Pasaje a modo protegido

Antes de pasar a modo protegido el sistema operativo necesita saber donde esta la gdt y cual es su tamaño para esto cargamos a traves de lgdt el registro gdtr el cual tiene esta información. También habilitamos A20 para permitir el acceso a direcciones superiores a  $2^{20}$  bits.

Luego realizamos un salto con la instruccion  $jmp\ 0x40:modoprotegido$  para poner a cs en el valor correcto, el 0x40 es para posicionarnos en el segmento de codigo de nivel 0 (indice 8 en la gdt). Una vez hecho esto estamos listos para pasar a modo protegido poniendo el bit PE de cr0 en 1.

Luego debemos establecer los selectores de segmentos de datos de nivel 0 y la base de la pila la cual se requirió que estuviera en la posición 0x27000

```
mov ax, 0x50
mov ds, ax
mov es, ax
mov gs, ax
mov fs, ax
mov ss, ax
mov ebp, 0x27000
mov esp, ebp
```

Hay un segmento mas que seteamos con el único objetivo de verificar el buen funcionamiento de la segmentación, este es  $GDT\_ID12\_SCREEN\_DESC$ , en la funcion  $pintar\_esquina\_superior\_izquierda$  definida en kernelasm se puede ver como accedemos al segmento y luego reescribimos los dos primeros pixels de la pantalla

## 2.2. Exceptiones

En esta sección explicaremos como realizamos la creación de la *Interrupt Descriptors Table* (IDT) y su posterior carga en el *kernel*.

#### 2.2.1. Inicialización de la IDT

La IDT es un arreglo de tamaño 255 de idt\_entry, la cual posee los campos a setear de una interrupt gate. Como en este apartado solo nos interesa asociar las excepciones internas del procesador, únicamente generamos las primeras 20 entradas de la tabla, siguiendo la tabla 6-1 del tercer volumen del manual de Intel <sup>1</sup>. Para inicializarla, utilizamos la función idt\_inicializar definida en idt.c, y cuya implementación se basa en llamar 20 veces al macro IDT\_ENTRY, definido en el mismo archivo, con los respectivos números de la excepción y el privilegio del descriptor (que en este caso querremos que sea siempre 0, pues es el kernel quien manejará estas excepciones).

La principal tarea de esta sección es, entonces, configurar correctamente la macro IDT\_ENTRY. También tendremos definidas 20 rutinas de atención para cada excepción, aunque de momento lo único que harán es mover el número de la excepción a eax y detener la ejecución del sistema. En el futuro, la rutina se encargará de desalojar la tarea que produjo la excepción y continuar con la ejecución. En la figura (1) mostramos como seteamos la entrada de la IDT correspondiente a la excepción n del procesador.

| 31                  | 16 | 15                               | 14 13 | 12 |   | 8                 | 7   | 5 | 4        | 0 |
|---------------------|----|----------------------------------|-------|----|---|-------------------|-----|---|----------|---|
| &_isrn[31:16]       |    | 1                                | 00    | 0  | 1 | 110               | 000 |   | 00000    |   |
| Offset (parte alta) |    | Р                                | DPL   | М  | D | Interrupt<br>gate | М   |   | Reserved |   |
| 31                  | 16 |                                  |       |    |   |                   |     |   |          | 0 |
| 0x40                |    | &_isrn[15:0] Offset (parte baja) |       |    |   |                   |     |   |          |   |
| Segment Selector    |    |                                  |       |    |   |                   |     |   |          |   |

Figura 1: Interrupt gate correspondiente a la excepción n del procesador.  $\_isrn$  es la rutina de atención de la interrupción n, mientras que  $\&\_isrn$  es la dirección donde se encuentra alojada la misma; el bit D en 1 indica que el gate es de 32bits. La M implica que el manual pide que esos bits estén seteados de esa forma.

Es importante destacar que el selector de segmento 0x40 corresponde a un segmento de código, pues recordemos que lo que esperamos encontrar en la dirección  $0x40:\&\_isrn$  es el código de un handler de interrupción. Además, el segmento requiere privilegio de kernel, lo cual es razonable pues esperamos que sea este quien resuelva las excepciones.

#### 2.2.2. Carga de la IDT

Para cargar el descriptor de la IDT (compuesto por su posición base y su tamaño) al IDTR usamos la operación lidt de la siguiente forma desde kernel.asm

lidt [IDT DESC]

## 2.3. Paginación básica

Aquí contaremos los pasos necesarios para poder activar la paginación y que esta esté configurada para realizar *identity mapping*. Este término se utiliza para describir un direccionamiento de memoria donde la dirección virtual coincide uno a uno con la física.

Las páginas que definimos nos mapean las direcciones 0x00000000 a 0x003FFFFF, el directorio de páginas se encuentra en la dirección 0x27000 y la tabla de páginas en 0x28000.

#### 2.3.1. Rutinas de mapeado y desmapeado

Escribimos dos rutinas para realizar la tarea, la de mapeado que nos permite asociar una dirección virtual con una física, y la de desmapeado que dada una dirección virtual la desasocia de memoria.

<sup>&</sup>lt;sup>1</sup>Intel 64 and IA-32 Architectures Software Developer's Manual, Volume 3: System programming guide, Chapter 6

#### Mapeado

La función mmu\_mapear\_pagina toma como parámetros la dirección virtual, el contenido del registro cr3, la dirección física y por último los atributos attrs.

1. Extraemos la dirección del directorio de tablas de cr3.

```
base directorio tablas = 0xFFFFF000 AND cr3
```

2. Calculamos el offset dentro del directorio de tablas en base a virtual.

```
offset directorio tablas = ((0xFFC00000 AND virtual) >> 22) << 2
```

3. Obtenemos el Page Directory Entry correspondiente.

4. Extraemos la dirección del directorio de páginas de pde.

```
base directorio paginas = 0xFFFFF000 AND pde
```

5. Calculamos el offset dentro del directorio de páginas en base a virtual.

offset directorio paginas = 
$$((0x003FF000 \text{ AND virtual}) >> 12) << 2$$

6. Obtenemos el puntero al Page Table Entry correspondiente.

```
ptr pte = base directorio paginas + offset directorio paginas
```

7. Asignamos al Page Table Entry la dirección de la física y los atributos attrs y forzamos que se limpien las tablas cacheadas para evitar la posibilidad de estar leyendo una entrada desactualizada.

```
*ptr_pte = (0xFFFFF000 AND fisica) OR attrs
Limpio cache de tablas
```

#### Desmapeado

La función mmu\_unmapear\_pagina toma como parámetros la dirección virtual y el contenido del registro cr3. Los primeros 6 pasos serán idénticos a los de mmu\_mapear\_pagina.

7. Seteamos en el Page Table Entry el bit de present en 0 y limpamos las tablas cacheadas para reflejar los cambios.

```
*ptr_pte = *ptr_pte AND 0xFFFFFFE
Limpio cache de tablas
```

### 2.3.2. Generación del directorio de páginas

Para mapear las direcciones solicitadas, desarrollamos la función mmu\_inicializar\_dir\_kernel que se encarga de generar la primer entrada en el directorio de tablas y su tabla de páginas asociada.

1. Primero creamos nuestro Page Directory Entry en la dirección 0x27000.

El número 0x3 equivale en binario a 11. Estos dos primeros bits encendidos corresponden dentro del Page Directory Entry a que el directorio de páginas estará presente y que además podrá ser leído y escrito.

```
*ptr pd = *ptr pd OR 0x28000 // Asignamos direccion
```

2. Llenamos el directorio de páginas aplicándole a cada entrada los mismos atributos descritos en el punto anterior.

#### 2.3.3. Activación de la paginación

Finalmente, teniendo las rutinas explicadas previamente, procedimos a activar la paginación en el procesador. El código siguiente corresponde a una sección del kernel.

1. Llamamos a nuestra rutina de inicialización del directorio de tablas

```
call mmu inicializar dir kernel
```

2. Cargamos la dirección del directorio de tablas en el registro cr3.

3. Activamos la paginación encendiendo el bit más significativo del registro cr0.

```
mov eax, cr0
or eax, 0x80000000
mov cr0, eax
```

## 2.4. Interrupciones externas

En la sección 2.2, cargamos las entradas de la IDT y generamos las rutinas de atención correspondientes a las excepciones internas del procesador. A continuación, haremos lo propio para las tres interrupciones externas que podrá recibir nuestro sistema: el reloj, el teclado, y una rutina encargada de proveer los servicios del sistema. Las dos primeras serán requeridas por el PIC, mientras que la última será para uso del usuario.

Dado que por defecto el mapeado del PIC entra en conflicto con las excepciones del procesador de la 8 a la 15, es necesario hacer previamente un remapeo. Para eso, la cátedra ya proporciona la función resetear\_pic que se encarga de remapear las *interrupt's requests* (IRQs) de reloj y de teclado a las interrupciones 32 (0x20) y 33 (0x21) respectivamente. Basta entonces llamar a esta función desde el kernel y luego reactivar el PIC con la función habilitar\_pic, también provista por la cátedra.

La interrupción para solicitar servicios del sistema será la 0x46.

Las entradas de la IDT se agregarán, al igual que antes, con el macro IDT\_ENTRY. Las de reloj y teclado tendrán DPL de kernel (0), mientras que para la 0x46 será de usuario (3).

A continuación explicamos que es lo que hace cada una de las rutinas de atención.

#### 2.4.1. Reloj

Esencialmente, el código de esta rutina es el mismo que el sugerido en la clase práctica de scheduler. La única diferencia real es que se agregan algunas líneas usadas por el modo debug.

Como primer paso, tras salvaguardar todos los registros para mantener la transparencia de la interrupción, llamamos a la función fin\_intr\_pic1 que se encarga de avisarle al PIC que el IRQ del reloj ya ha sido atendido, y por lo tanto el PIC nuevamente está disponible para recibir IRQs.

A continuación se llama a la función sched\_atender\_tick la cual se encargará de actualizar todo lo que sea necesario respecto del juego: posición del perro que está corriendo actualmente; removerlo del scheduler en caso de que deje de estar libre, ya sea porque volvió a la cucha y descargó todos sus huesos, o bien porque provocó una excepción; y finalmente setear la siguiente tarea a ejecutar, devolviendo el índice de su TSS en la GDT como output.

Como dicho output se recibe por el registro ax, se lo compara contra cx (al que previamente le cargamos el valor actual del task register con str) y en caso de ser iguales simplemente se sigue ejecutando la misma tarea. Caso contrario debe realizarse el task switching, haciendo un jump far con el selector de segmento correspondiente al tss de la siguiente tarea.

Finalmente, restauramos los registros con los valores que tenían antes de la interrupción.

- 2.4.2. Teclado
- 2.4.3. Servicios del sistema

## 2.5. Tareas

#### 2.5.1. Entradas de la GDT

Al llegar a esta parte del trabajo tuvimos que agregar 18 entradas nuevas a la GDT, estas describen las TSS's de tarea\_inicial, IDLE, los 8 perros posibles que puede tener el jugador A simultaneamente y lo mismo para el jugador B. Estos segmentos van del numero 13 al 30

Para configurar las entradas de la gdt se pone como base el 0, esto será luego seteado correctamente por  $tss\_inicializar$  la cual pone la dirección de la ubicación de la tss correspondiente, cada segmento tendrá como limite 104 Bytes (0x0067) para que la GDT interprete eso como bytes ponemos la granularidad en 0. El nivel de privilegio para las tareas de los jugadores será de 3. Las tareas que serán manejadas unicamente por el sistema,  $tarea\_inicial$  y IDLE tendran el DPL en 0. El bit de present va en 1 para todos los segmentos y todo lo demás quedara en 0.

#### 2.5.2. Inicialización de las TSS

Para completar las TSS's esta la función **tss\_inicializar**, esta como se dijo previamente pone la base correcta a los 18 segmentos descriptores de TSS's. En el codigo se puede ver el macro TSS\_PERRO\_ENTRY el cual esta definido en *tss.h* y realiza dicha operación de seteo de base para los segmentos de los perros.

Para la configuración de las opciones de la TSS IDLE pusimos la dirección de la pila en el mismo lugar que la del kernel (0x27000), aunque las pilas de menores privilegios, esp1 y esp2 estan en 0, cr3 vale igual que el cr3 del kernel, iomap vale 0xFFFF, y los selectores de segmentos se pusieronsegun correspondian. En eip ponemos la dirección de la tarea la cual se aclara en la consigna que es la 0x16000.

Una vez que esta todo configurado, inicializamos las tss y cargamos la tarea inicial en kernel.asm

```
; Cargar tarea inicial
mov ax, 0x68
ltr ax
```

#### 2.5.3. Construcción de Tareas

#### 2.5.4. La Interrupción 0x46

En nuestro sistema operativo la interrupción 0x46 es la rutina de atención de servicios, es ejecutada por los jugadores para decidir las acciones de los perros. Estas pueden variar entre game\_perro\_mover, game\_perro\_cavar, game\_perro\_olfatear y game\_perro\_recibir\_orden todos estos métodos estan definidos en perro.c por la catedra. Una vez hecha la acción si todavia el quantum no se acabo se llama a la tarea idle para que esta espere hasta el turno del siguiente jugador.

## 3. Resultados

# 4. Conclusiones