class: title, smokescreen, shelf, bottom, no-footer background-image: url(images/pcxt.jpg) # 181U Spring 2020 ### Software/Hardware Interface --- layout: true .footer[ - Geoffrey Brown, 2020 - 181U ] <style> h1 { border-bottom: 8px solid rgb(32,67,143); border-radius: 2px; width: 90%; } .smokescreen h1 { border-bottom: none; } .small.remark-slide-content.compact {font-size:1.2rem} .smaller.remark-slide-content.compact {font-size:1.1rem} .small-code.remark-slide-content.compact code {font-size:1.0rem} .very-small-code.remark-slide-content.compact code {font-size:0.9rem} .line-numbers{ /* Set "line-numbers-counter" to 0 */ counter-reset: line-numbers-counter; } .line-numbers .remark-code-line::before { /* Increment "line-numbers-counter" by 1 */ counter-increment: line-numbers-counter; content: counter(line-numbers-counter); text-align: right; width: 20px; border-right: 1px solid #aaa; display: inline-block; margin-right: 10px; padding: 0 5px; } </style> --- class: compact # Agenda * Memory Mapped I/O * Interrupts * DMA --- class: compact # Memory Mapped Input/Output (I/O) * Signals and Timing Diagrams * Functions and Latches * General Purpose I/O --- class: compact # Bus Model of Processor data:image/s3,"s3://crabby-images/ad215/ad2152cf848e97ef6034e98cc7f1f866140345e1" alt="" * CPU (central processing unit) communicates with memory and I/O devices over a bus. * Everything on the bus looks like memory to the CPU * A bus consists of a set of *shared* signal wires -- Address wires, Data wires, and Control wires. * Communication over bus is time dependent --- class: compact # Memory Operations At the machine instruction level, this interface is completely hidden. The Cortex-M (and most processors) has two basic memory access instructions. Load, which reads from "memory" with an address(pointer) ```asm ldr r1, [r2] @ r1 = *r2 ``` and Store, which writes to a memory address ```asm ldr r1, [r2] @ *r1 = r1 ``` --- class: compact # Cortex Address Map data:image/s3,"s3://crabby-images/f0b0b/f0b0b071aeb39406404b7483616889bfda3e73b9" alt="" The Cortex-M architecture reserves ranges of memory for specific functions -- * peripherals (hardware devices), * data (SRAM), * and code. --- class: compact # Aside on Timing Diagrams data:image/s3,"s3://crabby-images/3759f/3759f5baba3ebea02b4067ed6d34e86a1a0937ed" alt="" data:image/s3,"s3://crabby-images/42c56/42c568bc5307c311412585e8d2b0757b85417a13" alt="" * Timing diagrams provide a way to specify and reason about signals on a bus. * There are three electrical levels -- high, low, floating * There are two logical levels -- 1, 0 * We often describe groups of wires with hexadecimal or decoded values * Wires can be *driven* in both directions, but not at the same time --- class: compact # Timing Diagram Example -- Memory Access data:image/s3,"s3://crabby-images/3759f/3759f5baba3ebea02b4067ed6d34e86a1a0937ed" alt="" data:image/s3,"s3://crabby-images/421ec/421ecfe880950b53e17510f4c97b13b4d6b0dcb2" alt="" In this example there are three memory transactions * Read from address 0x04 -- returns 1 * Write 2 to address 0x04 * Read from address 0x04 -- returns 1 Notice that the Data lines are allowed to float between transactions. * On reads, the data lines are *driven* by the memory/device * On writes, the data lines are *driven* by the cpu --- class: compact # Hardware primitives -- finite functions data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/eaa85/eaa85f8f0a54a06f9c269dc9aa86a7bffec10aaf" alt="" * In principle, we can build any function with finite input and outputs from boolean logic gates * In practice, inputs and output signals are restricted to the processor word size (or less) * In practice, only relatively simple operations are implemented in a single functions (e.g. add,and).. ) --- class: compact # Hardware primitives -- latch data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/e9ffd/e9ffd191c8464de8a28c673d7b107eed303a535d" alt="" A *latch* provides a state-holding element. * We can build them in any width. * The basic operating is "load" which copies the input to the output * When a load is not being performed, the output is preserved even if the input changes. --- class: compact # Hardware primitives -- buffer data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/f887f/f887fd9d40250b315d603348479014f3af076af1" alt="" * A buffer provides a way to have multiple *drivers* on a bus * The output of a buffer *floats* when it is not enabled * When the buffer is enabled, its input is copied to its output --- class: compact # Example: A single memory word data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/b0de2/b0de224dad5af29e8241d2bb1fc071bbd270a1a9" alt="" --- class: compact # Example: Digital I/O Port data:image/s3,"s3://crabby-images/ba58d/ba58dfd4e9a8ec96a0f88478ce3459883b1d738b" alt="" data:image/s3,"s3://crabby-images/0fcc9/0fcc926b0729e48b05b801f6d815d8d87a7937cd" alt="" --- class: compact,hljs-tomorrow-night-eighties,line-numbers # C Model of I/O Port ```C typedef struct { volatile unsigned int in; volatile unsigned int out; volatile unsigned int dir; } port; ``` ```C extern port *p; p->dir = 3; // set bits (pins) 0 and 1 to outputs p->out = 1; // set pin 0 to 1 x = p->in; // read pins. ``` --- class: compact # Interrupts * Why Interrupts ? * Basic interrupt processing * Sources of interrupts * Interrupt handlers --- class: compact # Polling Suppose we are waiting for data from a communication device * Polling is the process of querying the device (through software) * Polling takes time away from other work * Polling makes it difficult to satisfy real-time constraints --- class: compact # Interrupts * **Interrupts** provide a general mechanism for hardware to signal the processor that attention is required * Rather than repeatedly asking devices if they need service (polling) let the devices signal the processor. --- class: compact # The big idea * For each type of event that potential needs service, associate a special procedure, called a **handler** that will be executed when the event occurs. * We call these events *interrupts* * In purely software implementations, **callbacks** serve the role of **handlers** -- this model is frequently used in GUIs * Interrupt **handlers** are called by the CPU * Before each "normal" instruction is executed, the CPU checks for pending events * If an event is pending, the CPU transfers control to the associated handler. --- class: compact # Interrupt Service data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/261b3/261b3e7f73aed254856f750aabbaa25b47df60ad" alt="" --- class: compact # Instruction Execution with Interrupt Handling data:image/s3,"s3://crabby-images/3381c/3381c5d68d92935bc3eab5cd22f652c7d8b6d3b4" alt="" data:image/s3,"s3://crabby-images/d2529/d2529ce0d0c6b5b6bb67a89dbcf66d64b6370cae" alt="" --- class: compact # Key Points for Programmers * If interrupts are **enabled**, they can occur at any point during program execution * Interrupts are **disabled** when the handler is called (at least from the same priority level) * Interrupt handlers are similar to normal procedures with a few caveats * They must save *everything*, not just the regular callee saved registers * They should not touch shared data except that specifically associated with the handler (e.g. an input queue) --- class: compact # The need to preserve registers (including status) data:image/s3,"s3://crabby-images/ba58d/ba58dfd4e9a8ec96a0f88478ce3459883b1d738b" alt="" data:image/s3,"s3://crabby-images/2ec31/2ec3142c4d3cb9bf72ad17e518b0c63002c948be" alt="" --- class: compact # Revised Memory Layout data:image/s3,"s3://crabby-images/ba58d/ba58dfd4e9a8ec96a0f88478ce3459883b1d738b" alt="" data:image/s3,"s3://crabby-images/906cc/906ccc8ccc56aaa2e570f1a6272bc7065c856fb0" alt="" --- class: compact # Sources of Interrupts * External device **interrupts** * timer * Serial port * Button * **Processor Exceptions** * Memory access violation * Divide by zero * **Software Interrupts** -- OS system calls --- class: compact,hljs-tomorrow-night-eighties,line-numbers # Direct Memory Access (DMA) * Example * What DMA does * STM32 DMA Architecture ```C for (i = 0; i < N; i++) { while(flagBusy); buf[i] = peripheralRegister; } ``` --- class: compact # Performance Without DMA data:image/s3,"s3://crabby-images/8ebee/8ebee2f3cba7ec796307554c2bbc212ecf7680f4" alt="" --- class: compact # Performance with DMA data:image/s3,"s3://crabby-images/882f7/882f77c2add94ffd43094d35dd1d7cd2cb2bd9e7" alt="" --- class: compact # How DMA works data:image/s3,"s3://crabby-images/3381c/3381c5d68d92935bc3eab5cd22f652c7d8b6d3b4" alt="" data:image/s3,"s3://crabby-images/7d505/7d505c54e9caffd77f02735523b6b72a5f99c70d" alt="" --- class: compact # STM32 DMA Architecture data:image/s3,"s3://crabby-images/3381c/3381c5d68d92935bc3eab5cd22f652c7d8b6d3b4" alt="" data:image/s3,"s3://crabby-images/83ee2/83ee277481ad2bb0657836833da6f2c9e3bd8b12" alt="" --- class: compact # Summary * Memory Mapped I/O * Interrupts * Direct Memory Acces * Cover (pxct): By <a href="https://de.wikipedia.org/wiki/User:Putput" class="extiw" title="de:User:Putput">Putput</a>, <a href="http://creativecommons.org/licenses/by-sa/3.0/" title="Creative Commons Attribution-Share Alike 3.0">CC BY-SA 3.0</a>, <a href="https://commons.wikimedia.org/w/index.php?curid=25662571">Link</a>