Download Núcleo de un Sistema Operativo

Document related concepts

Int 13h wikipedia , lookup

Interfaz de Paso de Mensajes wikipedia , lookup

Llamada de interrupción del BIOS wikipedia , lookup

Descriptor de archivo wikipedia , lookup

Int 16h wikipedia , lookup

Transcript
Departamento de Arquitectura y Tecnología de Computadores
Konputagailuen Arkitektura eta Teknologia Saila
_________________________________________
Laboratorio de Sistemas Operativos
_________________________________________
Núcleo de un Sistema
Operativo
Alberto Lafuente
Febrero 2006
Contenido
1
Introducción
2
Una estructura en capas para el sistema operativo
3
Estructura de un núcleo de sistema operativo
4
3.1
Rutinas dependientes del hardware
3.2
Rutinas de manejo de colas y auxiliares
3.3
Rutinas para la gestión de procesos
3.4
Definiciones y estructuras de datos
3.5
Primitivas del núcleo
Funcionamiento del núcleo
4.1
Gestión de procesos
4.2
Primitivas bloqueantes
4.3
Rutinas de tratamiento de interrupciones
4.4
Primitivas no bloqueantes
4.5
Primitivas de sincronización
5
Puesta en marcha
A.1
Prueba del núcleo
UPV/EHU ATC Laboratorio de Sistemas Operativos
2
1
Introducción
Un sistema operativo se define desde dos puntos de vista. En primer lugar, el sistema
operativo constituye la interfaz entre el usuario de un computador y los recursos de éste
(hardware y software), proporcionando una visión funcional del sistema en forma de llamadas
al sistema. En segundo lugar, el sistema operativo es el encargado de gestionar eficientemente
la utilización de los recursos por los usuarios.
Los servicios que un sistema operativo gestiona suelen dividirse en cuatro: procesador,
memoria, dispositivos y ficheros. La complejidad inherente a la gestión de alguno de estos
servicios hace necesario estructurar el sistema operativo en varias capas o niveles, cada una
ofreciendo un conjunto de primitivas a la inmediatamente superior. Por ejemplo, el sistema de
ficheros reside sobre el dispositivo disco, por lo que la gestión de ficheros se especificará en
base a las primitivas que proporcione la gestión del disco, que será la que programe el
hardware del dispositivo.
El nivel básico de un sistema operativo, que oculta las características hardware de la máquina,
se conoce como núcleo o kernel. En este documento se proporciona una descripción completa
de la estructura del núcleo de un sistema operativo multiprogramado, que incluye gestión de
procesos basada en prioridades, gestión de dispositivos (disco flexible, teclado, pantalla,
impresora, línea serie y reloj), y primitivas de sincronización (semáforos). Se ha escogido
como plataforma soporte la arquitectura PC basada en la familia i80x86. Ya que la mayor
parte de las características dependientes de la arquitectura están encapsuladas en un conjunto
de rutinas dependientes del hardware, esta elección no es especialmente determinante para el
diseño del sistema operativo. La difusión de esta arquitectura es la única razón para su
elección. Por otra parte, las rutinas dependientes del hardware proporcionan una interfaz C
para su uso en el núcleo, lo que facilita la portabilidad a otras plataformas.
2
Una estructura en capas para el sistema operativo
El esquema general propuesto es una estructura en tres niveles o capas, de abajo arriba:
1
Nivel Núcleo. Gestión básica de procesos: planificación a corto plazo, cambio de
contexto. Primitivas de sincronización. Gestión de E/S y tiempo, rutinas de atención.
2
Nivel del Sistema Básico de Ficheros, BFS. Sistema básico de ficheros: ubicación en
disco, directorios. Rutinas de E/S, servidores de dispositivos.
3
Nivel Sistema. Implementación de las llamadas al sistema; independencia del
dispositivo (tablas de canales); gestión de buffers para acceso a ficheros; carga,
ejecución y finalización de procesos.
La Figura 1 muestra la estructura en niveles de un sistema operativo. Para un nivel n sólo son
visibles las primitivas del nivel n-1. En general un nivel utilizará además una serie de rutinas
auxiliares internas que no son visibles desde el nivel superior.
3
Estructura de un núcleo de sistema operativo
Como ejemplo de núcleo de sistema operativo, en esta sección describiremos las rutinas
internas, estructuras de datos y primitivas de un núcleo multiprogramado para la arquitectura
UPV/EHU ATC Laboratorio de Sistemas Operativos
3
i80x86. Este núcleo es el que usaremos en el laboratorio. El funcionamiento del núcleo de
presentará en la siguiente sección.
Rutinas
de
Librería
USUARIO
NIVEL
SISTEMA
NIVEL
BFS
NIVEL
NUCLEO
Rutinas
internas
Nivel Sistema
Primitivas
Nivel Sistema
Rutinas
internas
Nivel BFS
Primitivas
Nivel BFS
Rutinas
internas
Nivel Núcleo
Primitivas
Nivel Núcleo
Rutinas
dependientes
del Hardware
HARDWARE
Figura 1. Niveles en un sistema operativo
3.1
Rutinas dependientes del hardware
Son las rutinas del núcleo de más bajo nivel. Distinguiremos dos grupos. Las más elementales
están escritas en ensamblador y tratan con direcciones específicas de E/S o de memoria (para
los dispositivos memory-mapped), o bien inhiben/activan interrupciones o manipulan
directamente la pila del programa para proporcionar los cambios de contexto. Sus prototipos
se describen en la Figura 2. El segundo grupo de rutinas internas del núcleo, que programan
los dispositivos, están codificadas en C. Estas se describen en la Figura 3.
Las rutinas dependientes del hardware se llamarán únicamente desde el núcleo. Encapsulan
las características de la arquitectura.
UPV/EHU ATC Laboratorio de Sistemas Operativos
4
RUTINAS BASICAS DEL NUCLEO
int flag inhibir ();
void desinhibir (int flag);
int in_port (int puerto);
void out_port (int puerto, int dato);
void strobe (int puerto, char mascara);
void eoi ();
void vector_int (int num_int , void (*rutina )());
int mascara_anterior enmascarar (int mascara);
void salvar_flags ();
void salvar_reg ();
void restaurar ();
void reset ();
void timer (int valor);
Figura 2. Rutinas dependientes del hardware en ensamblador
RELACIONADAS CON EL TERMINAL
char leer_teclado ();
void escribir_pantalla (int fila, int columna, char car, char atrib);
void posicionar_cursor (int fila , int columna);
void scroll (int fila_ini , int fila_fin);
int estado escribir_impresora (char car);
void beep (long tono_sonido);
RELACIONADAS CON LA LINEA SERIE
void ini_ls ();
int estado escribir_ls (char car);
char leer_ls ();
int estado consulta_ls_iir ();
int estado consulta_ls_lsr ();
int estado consulta_ls_msr ();
RELACIONADAS CON EL DISCO
void motor_on (int drive);
void motor_off ();
void posicionar_pista (int pista, int drive);
void programar_disco (int op, int cara, int pista, int sector, int drive);
int estado programar_dma (int operacion, char *buffer );
void comando_disco (int comando);
int estado comprobar_posicionado ();
int estado comprobar_trasferencia ();
int estado leer_status ();
void recalibrar (int drive);
Figura 3. Rutinas dependientes del hardware en C
3.2
Rutinas de manejo de colas y auxiliares
La mayoría se utilizan en el núcleo para el manejo de los PCBs. Fundamentalmente son
rutinas de acceso a colas, cuyos prototipos se describen en la Figura 4. Otras rutinas
auxiliares, de apoyo a la programación, se ilustran en la Figura 5. En general, las rutinas
auxiliares se utilizarán también en los niveles superiores. Por esta razón se definen las
estructuras item y cola de forma genérica, sin incluir información específica de los elementos
que enlazan (en general procesos).
UPV/EHU ATC Laboratorio de Sistemas Operativos
5
RUTINAS PARA MANEJO DE COLAS
struct item {
struct item *siguiente;
struct item *anterior;
int prioridad;
};
struct cola {
struct item *primero;
struct item *ultimo;
};
void inicializar_cola (struct cola *cola);
int vacia (struct cola *cola);
struct item *primero (struct cola *cola);
/* Elimina y devuelve el primer elemento de cola */
struct item *extraer (struct cola *cola, int prio);
/* Elimina y devuelve el primero de los de prioridad prio */
struct item *extirpar (struct cola *cola, struct item *elem);
/* Elimina y devuelve el elemento apuntado por elem */
void encolar (struct cola *cola, struct item *elem);
/* Encola elem al final de cola */
void insertar(struct cola *cola, struct item *elem, int prio);
/* Encola elem al final de los de prioridad prio */
void meter(struct cola *cola, struct item *elem, int estado);
/* Inserta elem actualizando estado en el PCB */
Figura 4. Rutinas auxiliares para manejo de colas
RUTINAS AUXILIARES DIVERSAS
void copiar (char *str1 , char *str2 , int lon);
int comparar (char *str1, char *str2 , int lon);
int buscar (char *str, char car, int lon);
void rellenar (char *str, char car, int lon);
void mayuscula (char *str, int lon);
void minuscula (char *str, int lon);
int long_a_ascii (long num, char *str, int base, int lon);
int int_a_ascii (int num, char *str, int base, int lon);
long ascii_a_long (char *str, int lon, int base);
int ascii_a_ int (char *str, int lon, int base);
Figura 5. Otras rutinas auxiliares
3.3
Rutinas para la gestión de procesos
La gestión de procesos está soportada por rutinas que manejan colas y proporcionan cambios
de estado en los procesos: bloquear(), scheduler(), dispatcher. Otras rutinas inicializan
estructuras de datos (PCB y pila) para los procesos en su creación. Estas rutinas se muestran
en la Figura 6 y se estudiarán más adelante.
RUTINAS PARA LA GESTION DE PROCESOS
void bloquear (struct cola *cola, int estado);
struct item *scheduler();
void dispatcher (struct item este_item)
void crear_pcb(void (*codigo)(), int *pila, int prio, int q, int id_proc));
void crear ();
Figura 6. Rutinas para la gestión de procesos
UPV/EHU ATC Laboratorio de Sistemas Operativos
6
3.4
Definiciones y estructuras de datos
Se refieren a la estructura del PCB y a las colas que modelan el funcionamiento del núcleo.
Aparecen en la Figura 7. Existe además una serie de definiciones de símbolos para los estados
de los procesos, dispositivos, etc, que no se muestran en la figura.
#define NUM_PCBS …
#define TAM_PILA …
/*
/*
numero maximo de PCB */
tamaño de la pila de un proceso
*/
struct pcb {
struct item elemento_cola;
int id_proceso;
int ax;
int dx;
int far *pila;
int quantum;
long cpu_time;
int status;
};
struct pcb proc[NUM_PCBS];
int pilas[NUM_PCBS][TAM_PILA];
struct info_proc {
int quantum;
int prioridad;
long t_cpu;
int status;
};
struct cola libres,
ready,
run,
teclado,
disco,
retardo,
lectura_ls,
escritura_ls;
int ntr = 0;
/* numero de ticks de reloj */
#define NSEM …
/* semáforos */
struct semaforo {
struct cola s;
int cont;
};
struct semaforo sem[NSEM];
Figura 7. Definiciones fundamentales del núcleo
Otro conjunto de definiciones (que tampoco se muestran aquí) especifican la configuración de
la máquina concreta (direcciones de pantalla, mapa de teclado, etc).
3.5
Primitivas del núcleo
Las primitivas del núcleo constituyen el conjunto de rutinas visibles desde el nivel superior.
Se definen en C y utilizan las rutinas dependientes del hardware y a las rutinas auxiliares
introducidas anteriormente. Las diferenciaremos denominándolas con el sufijo _nuc. Se
pueden dividir en los siguientes grupos:
UPV/EHU ATC Laboratorio de Sistemas Operativos
7
1
Control de dispositivos
(a) disco
void motor_on_nuc (int drive)
void motor_off_nuc ()
int posicionar_pista_nuc (int n_pista, int drive)
int leer_sector_nuc (int n_cara, int n_sector, char *p_buff)
int escribir_sector_nuc (int n_cara, int n_sector, char *p_buff)
int recalibrar_nuc (int drive)
(b) terminal
char leer_teclado_nuc ()
int escribir_pantalla_nuc (int lin, int col, char car, char atributo)
int scroll_nuc (int lin_sup, int lin_inf)
(c) línea serie
int leer_l_s_nuc ()
void escribir_l_s_nuc (char car)
void init_l_s_nuc ()
(d) impresora
int escribir_impresora_nuc (char car)
2
Control de procesos
int crear_pcb_nuc (void (*cod)(), int *pila, int prio, int quantum, int pid)
int destruir_pcb_nuc (int id_proc)
int quisoc_nuc ()
int info_proc_nuc (int id_proceso, struct info_proc *p_info)
int modif_proc_nuc (int id_proceso, struct info_proc *p_info)
3
Espera por tiempo
void retardo_nuc (int n_tics)
4
Sincronización entre procesos
int wait_nuc (int n_sem)
int signal_nuc (int n_sem)
int init_sem_nuc (int n_sem, int valor_inicial)
5
Reset
void reset_nuc ()
UPV/EHU ATC Laboratorio de Sistemas Operativos
8
4
Funcionamiento del núcleo
4.1
Gestión de procesos
Incluiremos aquí planificación, creación y destrucción de procesos. Un proceso se representa
en un PCB (struct pcb) que se identifica por un entero (índice en la tabla de PCBs).
3.4.1 Planificación de procesos
En la versión del núcleo propuesta, se proporciona planificación de procesos de tiempo
compartido y prioridades, con expulsión por desbloqueo de un proceso. Esto implica que la
gestión de procesos involucra a las rutinas de atención (muy en particular a la de reloj para
control de fin de quantum), además de las que hacen que un proceso pase a estado bloqueado
(más adelante nos referiremos a estas rutinas como bloqueantes).
Los procesos preparados para ejecución se encolan en la cola ready de acuerdo a su prioridad,
por lo que la rutina del scheduler se limita a coger el primer elemento de la cola, que pasa al
dispatcher para que lo cargue en ejecución (cola run, que tendrá un único elemento). Por
ortogonalidad, se asume la existencia de un proceso nulo, que siempre estará preparado para
ejecución o ejecutándose. La Figura 8 muestra el código de las rutinas scheduler() y el
dispatcher(). La rutina bloquear() es la encargada de sacar a un proceso de la cola, lo que
implica como efecto colateral la actualización del tiempo de CPU. Su código se muestra en la
Figura 9.
struct item *scheduler()
{
return (primero(&ready));
}
void dispatcher (este_item)
struct item *este_item;
{
meter (&run, este_item, RUN);
}
Figura 8. Rutinas scheduler y dispatcher
void bloquear (c, status)
struct cola *c;
int status;
{
((struct pcb *)run.primero)->cpu_time += ntr;
meter(c, primero(&run), status);
ntr=0;
}
Figura 9. Rutina bloquear
Una vez actualizadas la colas de procesos, la multiplexación se completa reestableciendo el
estado del procesador para que ejecute el proceso planificado. Básicamente esto implica hacer
que el SP apunte a la pila del nuevo proceso. La rutina restaurar(), codificada en
ensamblador, es la que se encarga de ello, manipulando el bloque de activación anterior.
Previamente, la rutina salvar_reg() habrá salvado el estado del proceso que estaba en
ejecución.
UPV/EHU ATC Laboratorio de Sistemas Operativos
9
La gestión del tiempo compartido la realiza la rutina de atención al reloj, multiplexar(), como
se ilustra en la Figura 10. Esta rutina incluye otras funciones (apagado automático del motor
de la unidad de disco y control del periodo transitorio tras el encendido, retardo de procesos
por tiempo), que no aparecen en la figura.
void multiplexar ()
{
salvar_reg();
/* FIN DE QUANTUM: */
if (++ntr==((struct pcb*)run.primero)->quantum) cambiar = TRUE;
…
if (cambiar) {
bloquear (&ready, READY);
dispatcher (scheduler());
}
eoi();
restaurar();
}
Figura 10. Rutina multiplexar. Control de quantum.
3.4.2 Creación, control y destrucción de procesos
La rutina crear_pcb_nuc() asigna un PCB para un nuevo proceso, provocando la expulsión
del que está en ejecución. Existen primitivas para extraer información de un proceso y su
identificador (info_proc_nuc(), quisoc_nuc()) y para modificar su quantum y prioridad
(modif_proc_nuc()). Finalmente, en la Figura 11 se incluye el código de
destruir_pcb_nuc(). Nótese las situaciones en que el PCB no se libera (el proceso tiene una
petición pendiente), sino que el proceso queda en estado finalizado (DEAD). La rutina de
atención correspondiente liberará el PCB del proceso DEAD con la petición pendiente.
4.2
Primitivas bloqueantes
Llamaremos rutinas bloqueantes del núcleo a las que solicitan un servicio del hardware (un
dispositivo, el reloj…) y dejan al proceso en estado bloqueado, provocando la planificación
de un nuevo proceso. Son varias las primitivas que trabajan de esta forma, en general aquéllas
cuyo servicio se atenderá mediante interrupciones, como motor_on_nuc(), retardo_nuc() —
por tiempo—; leer_teclado_nuc() —por teclado—; escribir_impresora_nuc() —por
impresora— leer_l_s_nuc, escribir_l_s_nuc() —por línea serie—; posicionar_pista_nuc(),
leer_sector_nuc(), escribir_sector_nuc(), y recalibrar_nuc() —por disco—. Como
ejemplo, en la Figura 12 se muestra el código de leer_teclado_nuc(). Todas las rutinas
bloqueantes siguen el protocolo de salvar el contexto del proceso (salvar_flags y
salvar_reg), bloquear al proceso y planificar otro proceso cargando su contexto. Nótese
también que se exige que la cola de bloqueado esté vacía: se sugiere la implementación de un
esquema cliente-servidor en el nivel superior.
UPV/EHU ATC Laboratorio de Sistemas Operativos
10
int destruir_pcb_nuc (id_proceso)
int id_proceso;
{
inhibir();
salvar_flags();
salvar_reg();
id_proceso = id_proceso % NUM_PCBS;
switch (proc[id_proceso].status) {
case LIBRE:
((struct pcb *)run.primero)->ax = -1;
break;
case READY:
proc[id_proceso].status = LIBRE;
encolar (&libres, extirpar(&ready,
&(proc[id_proceso].elemento_cola)));
((struct pcb *)run.primero)->ax = 0;
break;
case RUN:
proc[id_proceso].status = LIBRE;
encolar (&libres,primero (&run));
dispatcher (scheduler());
break;
case BLOQ_TEC :
case BLOQ_DISC:
case BLOQ_RET:
proc[id_proceso].status = DEAD;
((struct pcb *)run.primero)->ax = 0;
break;
default:
proc[id_proceso].status = LIBRE;
encolar(&libres, extirpar (&sem[proc[id_proceso].status].s,
&(proc[id_proceso].elemento_cola)));
((struct pcb *)run.primero)->ax = 0;
break;
}
restaurar();
}
Figura 11. Rutina destruir_pcb_nuc
char leer_teclado_nuc ()
{
inhibir();
salvar_flags();
salvar_reg();
if (vacia (&teclado)) {
bloquear(&teclado, BLOQ_TEC);
dispatcher (scheduler());
restaurar();
}
else crash();
}
Figura 12. Ejemplo de rutina bloqueante
4.3
Rutinas de tratamiento de interrupciones
Aunque la rutina multiplexar puede servir como ejemplo de rutina de tratamiento de
interrupciones, resultará clarificador en este momento introducir el código de la rutina de
tratamiento de las interrupciones de teclado (Figura 13) para completar el ejemplo de
tratamiento de peticiones del apartado anterior. Hay que resaltar alguna cuestión importante,
común al resto de rutinas de este tipo:
UPV/EHU ATC Laboratorio de Sistemas Operativos
11
•
Tratamiento de peticiones pendientes de pocesos finalizados. La rutina de atención
detecta que el proceso está en estado DEAD (véase destruir_pcb_nuc()) y libera su
PCB.
•
Valores de retorno. El carácter leído (o un código de error en otras rutinas) debe ser
devuelto a través del PCB del proceso (campo ax), que será lo que entregue como
valor de retorno la primitiva que produjo el bloqueo (leer_teclado en nuestro ejemplo)
al restaurarse el contexto del proceso (rutina restaurar()) cuando éste sea planificado
nuevamente.
void int_teclado()
{
salvar_reg();
if ((c=leer_teclado()) != -1)
if (!vacia(&teclado))
if (((struct pcb *)teclado.primero)->status == DEAD) {
((struct pcb *)teclado.primero)->status= LIBRE;
encolar(&libres, primero(&teclado));
}
else {
((struct pcb *)teclado.primero)->ax= c;
meter(&ready, primero(&teclado), READY);
bloquear(&ready, READY);
dispatcher (scheduler());
}
else beep();
eoi();
restaurar();
}
Figura 13. Ejemplo de rutina de tratamiento de interrupciones
4.4
Primitivas no bloqueantes
Llamaremos rutinas no bloqueantes a las que no dejan al proceso bloqueado. En general serán
de este tipo aquellas primitivas que no se sirven mediante interrupciones, como es el caso de
esribir_pantalla_nuc(), que se muestra en la Figura 14. Otras primitivas de este tipo son:
scroll_nuc(), motor_off_nuc(), quisoc_nuc() e info_proc_nuc(). Estas primitivas se
resuelven como simples llamadas a funciones y no producen cambios de contexto.
int escribir_pantalla_nuc (linea, columna, caracter, atributo)
int linea, columna;
char caracter, atributo;
{
if (0<=linea && linea<=24 && 0<=columna && columna<=79 &&
(atributo==UNDERSCORE || atributo==INVERSE ||
atributo==BLINK || atributo==NORMAL)) {
escribir_pantalla (linea, columna, caracter, atributo);
return (0);
}
else return (-1);
}
Figura 14. Ejemplo de rutina no bloqueante
UPV/EHU ATC Laboratorio de Sistemas Operativos
12
4.5
Sincronización entre procesos
El núcleo propuesto ofrece semáforos como mecanismo básico de sincronización entre
procesos. Un semáforo se identifica por un entero. Se introduce una nueva condición de
bloqueo para los procesos, con tantas colas como semáforos disponibles, NSEM. El bloqueo
por semáforo presenta la particularidad de que debe seguir estrictamente una disciplina FCFS,
por lo que el mecanismo de bloqueo estándar introducido para las rutinas bloqueantes no se
puede aplicar en la primitiva de espera por semáforo, wait_nuc(), descrita en la Figura 15. La
primitiva signal_nuc() se ha implementado como expulsora.
int wait_nuc (n_semaforo)
int n_semaforo;
{
inhibir();
salvar_flags();
salvar_reg();
if (0<=n_semaforo && n_semaforo<NSEM) {
((struct pcb *)run.primero)->ax = 0;
if (sem[n_semaforo].cont > 0) sem[n_semaforo].cont--;
else {
((struct pcb *)run.primero)->status = n_semaforo;
((struct pcb *)run.primero)->cpu_time += ntr;
encolar (&sem[n_semaforo].s, primero(&run));
ntr=0;
dispatcher (scheduler());
}
}
else ((struct pcb *)run.primero)->ax = -1;
restaurar();
}
Figura 15. Rutina wait_nuc
5
Puesta en marcha
Sobre el núcleo propuesto se puede construir un sistema operativo completo. Como no
disponemos de un sistema operativo completo con todas sus herramientas para soportar
aplicaciones, probaremos el funcionamiento del núcleo con un conjunto de programas de
prueba que utilicen sus primitivas y monitoricen su funcionamiento. En cualquier caso, se
requiere un programa que instale el núcleo y lance los procesos del sistema. Este programa
incluirá:
•
El código completo del núcleo, que incluye primitivas, rutinas internas y librerías de
bajo nivel.
•
El código de la prueba del núcleo, que es un conjunto de funciones en código C que se
ejecutarán asíncronamente, como threads o procesos a ser gestionados por el núcleo.
•
El procedimiento principal (main) de puesta en marcha.
El procedimiento principal realizará las siguientes acciones:
(1) Instalación de las rutinas de tratamiento de interrupciones, mediante vector_int().
(2) Inicialización de las colas del núcleo (run, ready, teclado, semáforos, …).
UPV/EHU ATC Laboratorio de Sistemas Operativos
13
(3) Inicialización de los PCBs como libres, encolándolos en la cola de libres.
(4) Creación de los procesos del sistema. Al menos son necesarios dos: el proceso nulo, y
un proceso de inicialización, que lanzará el resto ya sobre el núcleo. Estos procesos se
crean rellenando directamente los campos de los PCBs y utilizando la rutina
crear_pcb() para asignar la pila correspondiente y apuntar al código del proceso.
(5) Planificar (dispatcher y scheduler) y cargar el estado del proceso seleccionado
(mediante restaurar()). Este proceso será el de inicialización, ya que el nulo se habrá
creado con la prioridad mínima.
Ya que no disponemos de un dispositivo de arranque propio, la puesta en marcha del núcleo
requiere de un sistema operativo anfitrión, MS-DOS. El programa de prueba permite volver a
MS-DOS (al menos si no se provocan situaciones de fallo que requieran reiniciar el sistema).
Naturalmente, el núcleo puede ser lanzado desde cualquier sistema operativo Windows 95 o
98, lo que facilita el desarrollo y manejo del software, pero no desde un sistema Windows
basado en NT (2000 o XP). En este último caso, aún se puede recurrir a un emulador
(VMware o Virtual PC) que soporte MS-DOS.
UPV/EHU ATC Laboratorio de Sistemas Operativos
14
A1 Prueba del núcleo
A1.1 Compilación y montaje
Los ficheros fuente del núcleo que se proporcionan son:
rutinas.h
Definiciones fundamentales del núcleo; declaraciones de rutinas auxiliares y
de manejo de colas, y declaraciones de rutinas ensamblador y C
dependientes del hardware.
maquina.h
Definiciones (condicionales) dependientes de la pantalla y el teclado.
nucleo.h
Primitivas del núcleo.
Para probar esta versión del núcleo, se proporciona también un programa de prueba,
pruebas.h (que hace las veces de nivel superior). pruebas.h se incluye junto a nucleo.h en un
fichero fuente fase0.c, que contiene el código para poner en marcha el sistema. Un resumen
de la organización de los módulos fuente puede verse en la Figura A1.1.
Se proporciona un comando para compilar el código:
ccc nombre_programa
Para la prueba de la versión propuesta, nombre_programa será fase0
Se proporciona otro comando para montar el código obtenido con los módulos objeto
proporcionados, de acuerdo al esquema de la Figura A1.2:
li nombre_programa
A1.2 Ejecución
Una vez compilado y montado el código, se puede probar el funcionamiento del núcleo con
ayuda del código ejecutable obtenido. Al ejecutar fase0 se ofrecen mediante un menú las
pruebas de diferentes aspectos del núcleo:
•
Control de procesos. Permite manejar prioridades y quanta de procesos.
•
Control del disco flexible. Permite operar el motor, posicionar el cabezal y
transferir información entre disco y memoria.
La introducción de otras opciones se realizará modificando el código de pruebas.h.
UPV/EHU ATC Laboratorio de Sistemas Operativos
15
rutinas.h
nucleo.h
Rutinas
auxiliares
Definiciones,
rutinas
auxiliares,
Definiciones
etc
Rutinas ensambador
(namejo del Hw)
fase0.c
#include "rutinas.h"
#include "nucleo.h"
#include "maquina.h"
#include "pruebas.h"
Código de las
primitivas del núcleo
Programas que
usan el núcleo
Programa de
activación del núcleo
Rutinas dependientes
de la máquina
maquina.h
pruebas.h
Figura A1.1. Organización de los módulos fuente
fas e0. c
fas e0. obj
nucleo.h
pruebas.h
main()
…
ccc fase0
li fase0
fas e0. exe
s olib. lib
Figura A1.2. Esquema para compilar y montar el núcleo.
UPV/EHU ATC Laboratorio de Sistemas Operativos
16