Skip to content

Latest commit

 

History

History
173 lines (121 loc) · 9.2 KB

README.ES.md

File metadata and controls

173 lines (121 loc) · 9.2 KB

Tutoriales de desarrollo de Sistemas Operativos en Rust con la Raspberry Pi


ℹ️ Introducción

Esto es una serie de tutoriales para los desarrolladores aficionados a los Sistemas Operativos (OS) que se están adentrando a la nueva arquitectura ARM de 64 bits ARMv8-A architecture. Los tutoriales darán una guía paso a paso en cómo escribir un Sistema Operativo monolítico desde cero. Estos tutoriales cubren la implementación común de diferentes tareas de Sistemas Operativos, como escribir en una terminal serie, configurar la memoria virtual y manejar excepciones de hardware (HW). Todo mientras usamos la seguridad y velocidad que Rust nos proporciona.

¡Divértanse!

Atentamente,
Andre (@andre-richter)

P.S.: Para otros lenguajes, por favor busquen los diferentes archivos README. Por ejemplo, README.CN.md o README.ES.md. Muchas gracias a nuestros traductores 🙌.

📑 Estructura

  • Cada tutorial contiene un solo binario arrancable correspondiente al núcleo.
  • Cada tutorial nuevo extiende el tutorial anterior.
  • Cada tutorial tendrá un README y cada README tendrá un pequeña sección de tl;dr en donde se dará una pequeña perspectiva general de los cambios y se mostrará el código fuente diff del tutorial anterior para que se puedan inspeccionar los cambios/adiciones que han ocurrido.
    • Algunos tutoriales además de tener un tl;dr también tendrán una sección en la que se dará una explicación con todo lujo de detalle. El plan a largo plazo es que cada tutorial tenga una buena explicación además del tl;dr y el diff; pero por el momento los únicos tutoriales que gozan de una son los tutoriales en los que creo que el tl;dr y el diff no son suficientes para comprender lo que está pasando.
  • El código que se escribió en este tutorial soporta y corre en la Raspberry Pi 3 y en la Raspberry 4
    • Del tutorial 1 hasta el 5 son tutoriales "preparatorios", por lo que este código solo tendrá sentido ejecutarlo en QEMU.
    • Cuando llegues al tutorial 5 podrás comenzar a cargar y a ejecutar el núcleo en una Raspeberry de verdad, y observar la salida serie (UART).
  • Aunque la Raspberry Pi 3 y 4 son las principales placas este código está escrito en un estilo modular, lo que permite una fácil portabilidad a otras arquitecturas de CPU o/y placas.
    • Me encantaría si alguien intentase adaptar este código en una arquitectura RISC-V.
  • Para la edición recomiendo Visual Studio Code con Rust Analyzer.
  • En adición al texto que aparece en los tutoriales también sería recomendable revisar el comando make doc en cada tutorial. Este comando te deja navegar el código documentado de una manera cómoda.

Salida del comando make doc

make doc

🛠 Requisitos del sistema

Estos tutoriales están dirigidos principalmente a distribuciones de Linux. Muchas de las cosas vistas aquí también funcionan en macOS, pero esto solo es experimental.

🚀 La versión tl;dr

  1. Instala Docker Desktop.

  2. (Solo para Linux) Asegúrate de que la cuenta de tu usuario está en el grupo docker.

  3. Prepara la Rust toolchain. La mayor parte se hará automáticamente durante el primer uso del archivo rust-toolchain.toml. Todo lo que nos queda hacer a nosotros es:

    i. Si ya tienes una versión de Rust instalada:

    cargo install cargo-binutils rustfilt

    ii. Si necesitas instalar Rust desde cero:

    curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh
    
    source $HOME/.cargo/env
    cargo install cargo-binutils rustfilt
  4. En caso de que uses Visual Studio Code, recomiendo que instales la extensión Rust Analyzer extension.

  5. (Solo para macOS) Instala algunas Ruby gems.

    Ejecuta esto en la carpeta root del repositorio:

    bundle install --path .vendor/bundle --without development

🧰 Más detalles: Eliminando Lios con Toolchains

Esta serie trata de enfocarse lo máximo posible en tener una experiencia agradable para el usuario. Por lo tanto, se han dirigido muchos esfuerzos a eliminar la parte más difícil del desarrollo de los sistemas incorporados (embedded) tanto como se pudo.

Rust por sí mismo ya ayuda mucho, porque tiene integrado el soporte para compilación cruzada. Todo lo que necesitamos para compilar desde una máquina con una arquitectura x86 a una Raspberry Pi con arquitectura AArch64 será automáticamente instalado por rustup. Sin embargo, además de usar el compilador de Rust, también usaremos algunas otras herramientas, entre las cuales están:

  • QEMU para emular nuestro núcleo en nuestra máquina principal.
  • Una herramienta llamada Minipush para cargar el núcleo en una Raspberry Pi cuando queramos usando UART.
  • OpenOCD y GDB para hacer depuración ("debugging") en la máquina a instalar.

Hay muchas cosas que pueden salir mal mientras instalamos y/o compilamos las versiones correctas de cada herramienta en tu máquina. Por ejemplo, tu distribución de Linux tal vez podría no proporcionar las versiones más recientes de paquetes que se necesiten. O tal vez te falten algunas dependencias para la compilar estas herramientas.

Esta es la razón por la cual usaremos Docker mientras sea posible. Te estamos proporcionando un contenedor que tiene todas las herramientas o dependencias preinstaladas. Si quieres saber más acerca de Docker y revisar el contenedor proporcionado, por favor revisa la carpeta docker del repositorio.

📟 Puerto Serie USB

Ya que el núcleo desarrollado en este tutorial se ejecuta en hardware real, se recomienda que tengas un adaptador de puerto serie USB cable para sentir la experiencia completa.

  • Puedes encontrar estos cables que deberían funcionar sin ningún problema en [1] [2], pero hay muchos otros que pueden funcionar. Idealmente, tu cable estaría basado en el chip CP2102.
  • Lo conectas a los pines GND y GPIO 14/15 como se muestra en la parte inferior.
  • Tutorial 5 es la primera vez en la que lo vas usar. Revisa las instrucciones sobre cómo preparar una tarjeta SD para arrancar en tu núcleo desde ahí.
  • Empezando con el tutorial 6, arrancar núcleos en tu Raspberry Pi comienza a ser más fácil. En este tutorial se desarrolla un chainloader, que será el último archivo que necesitarás copiar de manera manual a la tarjeta SD por el momento. Esto te permitirá cargar los núcleos de los tutoriales durante el arranque usando UART.

UART wiring diagram

🙌 Agradecimientos

La versión original de estos tutoriales empezó como un fork de los increíbles tutoriales de programación en hardware en la RPi3 en C de Zoltan Baldaszti. ¡Gracias por darme un punto de partida!

Traducciones de este repositorio

  • Chino:
  • Español:
    • @zanezhub.
    • En el futuro habrán tutoriales traducidos al español.

Licencia

Este proyecto está licenciado por cualquiera de las siguientes licencias como alguna de tus dos opciones

Contribución

A menos de que lo menciones, cualquier contribución enviada por ti para su inclusión en este trabajo, tal como se define en la licencia Apache-2.0, deberá tener doble licencia como se muestra en la parte superior, sin ningún cambio de términos o condiciones.