Chapter 09 Embedded Firmware Design and Development
Chapter 09 Embedded Firmware Design and Development
 The embedded firmware is responsible for controlling the various peripherals of the
  embedded hardware and generating response in accordance with the functional
  requirements of the product
 The embedded firmware is usually stored in a permanent memory (ROM) and it is non
  alterable by end users
 Designing Embedded firmware requires understanding of the particular embedded
  product hardware, like various component interfacing, memory map details, I/O port
  details, configuration and register details of various hardware chips used and some
  programming language (either low level Assembly Language or High level language like
  C/C++ or a combination of the two)
 The embedded firmware development process starts with the conversion of the firmware
  requirements into a program model using various modeling tools
 There exist two basic approaches for the design and implementation of embedded
  firmware, namely;
      The Super loop based approach
      The Embedded Operating System based approach
 The decision on which approach needs to be adopted for firmware development is purely
  dependent on the complexity and system requirements
                                                                                     2
               Embedded Firmware Design & Development
                                                                                                 4
             Embedded Firmware Design & Development
Embedded firmware Design Approaches – Embedded OS based
Approach
 The embedded device contains an Embedded Operating System which can be one of:
     A Real Time Operating System (RTOS)
     A Customized General Purpose Operating System (GPOS)
 The Embedded OS is responsible for scheduling the execution of user tasks and the
  allocation of system resources among multiple tasks
 Involves lot of OS related overheads apart from managing and executing user defined
  tasks
 Microsoft® Windows XP Embedded is an example of GPOS for embedded devices
 Point of Sale (PoS) terminals, Gaming Stations, Tablet PCs etc are examples of
  embedded devices running on embedded GPOSs
 ‘Windows CE’, ‘Windows Mobile’,‘QNX’, ‘VxWorks’, ‘ThreadX’, ‘MicroC/OS-II’,
  ‘Embedded Linux’, ‘Symbian’ etc are examples of RTOSs employed in Embedded
  Product development
 Mobile Phones, PDAs, Flight Control Systems etc are examples of embedded devices
  that runs on RTOSs                                                                5
         Embedded Firmware Design & Development
Assembly Language
High Level Language
  Subset of C (Embedded C)
  Subset of C++ (Embedded C++)
  Any other high level language with supported Cross-compiler
Mix of Assembly & High level Language
  Mixing High Level Language (Like C) with Assembly Code
  Mixing Assembly code with High Level Language (Like C)
  Inline Assembly
                                                                 6
              Embedded Firmware Design & Development
                                                                                               7
                Embedded Firmware Design & Development
;####################################################################
;       SUBROUTINE FOR GENERATING DELAY
;       DELAY PARAMETR PASSED THROUGH REGISTER R1
;       RETURN VALUE NONE
;       REGISTERS USED: R0, R1
;####################################################################
        DELAY: MOV      R0, #255 ; Load Register R0 with 255
                DJNZ    R1, DELAY         ; Decrement R1 and loop till R1= 0
                RET                       ; Return to calling program
The symbol ; represents the start of a comment. Assembler ignores the text in a line after the ;
 symbol while assembling the program
DELAY is a label for representing the start address of the memory location where the piece of code
 is located in code memory
The above piece of code can be executed by giving the label DELAY as part of the instruction. E.g.
 LCALL DELAY; LMP DELAY
                                                                                                  9
                 Embedded Firmware Design & Development
Drawbacks:
                                                                   12
           Embedded Firmware Design & Development
          Source File 1
                                          Module
          (.c /.c++ etc)                                        Object File 1
                                       Cross-compiler
           (Module-1)
          Source File 2
                                          Module
          (.c /.c++ etc)                                        Object File 2
                                       Cross-compiler
           (Module-2)
          Machine Code
           (Hex File)
                             High level language to machine language Conversion process
                                                                                          14
        Embedded Firmware Design & Development
                                                                15
          Embedded Firmware Design & Development
                                                                          16
                Embedded Firmware Design & Development
High Level Language – ‘C’ V/s Embedded C
 ‘C’ is a well structured, well defined and standardized general purpose programming language with
  extensive bit manipulation support
 ‘C’ offers a combination of the features of high level language and assembly and helps in hardware
  access programming (system level programming) as well as business package developments
  (Application developments like pay roll systems, banking applications etc)
 The conventional ‘C’ language follows ANSI standard and it incorporates various library files for
  different operating systems
 A platform (Operating System) specific application, known as, compiler is used for the conversion
  of programs written in ‘C’ to the target processor (on which the OS is running) specific binary files
 Embedded C can be considered as a subset of conventional ‘C’ language
 Embedded C supports all ‘C’ instructions and incorporates a few target processor specific
  functions/instructions
 The standard ANSI ‘C’ library implementation is always tailored to the target processor/controller
  library files in Embedded C
 The implementation of target processor/controller specific functions/instructions depends upon the
  processor/controller as well as the supported cross-compiler for the particular Embedded C
  language
 A software program called ‘Cross-compiler’ is used for the conversion of programs written in
                                                                                                       17
  Embedded C to target processor/controller specific instructions
               Embedded Firmware Design & Development
High Level Language Based Development – ‘Compiler’ V/s ‘Cross-
Compiler’
 Compiler is a software tool that converts a source code written in a high level language on top of a
  particular operating system running on a specific target processor architecture (E.g. Intel
  x86/Pentium).
 The operating system, the compiler program and the application making use of the source code run
  on the same target processor.
 The source code is converted to the target processor specific machine instructions
 A native compiler generates machine code for the same machine (processor) on which it is
  running.
 Cross compiler is the software tools used in cross-platform development applications
 In cross-platform development, the compiler running on a particular target processor/OS converts
  the source code to machine code for a target processor whose architecture and instruction set is
  different from the processor on which the compiler is running or for an operating system which is
  different from the current development environment OS
 Embedded system development is a typical example for cross-platform development where
  embedded firmware is developed on a machine with Intel/AMD or any other target processors and
  the same is converted into machine code for any other target processor architecture (E.g. 8051,
  PIC, ARM9 etc).
 Keil C51compiler from Keil software is an example for cross-compiler for 8051 family                   18
  architecture
      Embedded Firmware Design & Development
                    Programming in Embedded C
Keywords
auto       double          int              struct
break      else            long             switch
case       enum            register         typedef
char       extern          return           union
const      float           short            unsigned
continue   for             signed           void
default    goto            sizeof           volatile
do         if              static           while
                                                       19
          Embedded Firmware Design & Development
                               Programming in Embedded C
Logical Operations
Operator      Operation        Comments
                               Performs logical AND operation. Output is true (logic 1) if
&&            Logical AND      both operands (left to and right to of && operator) are true
                               Performs logical OR operation. Output is true (logic 1) if
||            Logical OR       either operand (operands to left or right of || operator) is
                               true
!             Logical NOT      Performs logical Negation. Operand is complemented
                               (logic 0 becomes 1 and vice versa)
                                                                                              22
                  Embedded Firmware Design & Development
                        Programming in Embedded C
Branching Instructions
Conditional branching Explanation
Instruction
//if statement
if (expression)           Evaluates the expression first and if it is true executes the statements given within the { }
{                         braces and continue execution of statements following the closing curly brace (}). Skips
statement1;               the execution of the statements within the curly brace { } if the expression is false and
statement2;               continue execution of the statements following the closing curly brace (}).
………….;                    One way branching
}
statement 3;
…………..;
//if else statement
if (expression)           Evaluates the expression first and if it is true executes the statements given within the { }
{                         braces following if (expression) and continue execution of the statements following the
if_statement1;            closing curly brace (}) of else block. Executes the statements within the curly brace { }
if_statement2;            following the else, if the expression is false and continue execution of statements
…………….;                   following the closing curly brace (}) of else.
}                         Two way branching
else
{
else_statement1;
else_statement2;
……………….;
}
statement 3;                                                                                                              23
                  Embedded Firmware Design & Development
                                     Programming in Embedded C
Branching Instructions
//exiting from loop                      Loops can be exited in two ways. First one is normal exit where loop is exited when the
break;                                   expression/test for condition becomes false. Second one is forced exit. break and goto
goto label                               statements are used for forced exit.
                                         break exits from the inner most loop in a deeply nested loop, whereas goto transfers the
                                         program flow to a defined label.
                                                                                                                                        25
                Embedded Firmware Design & Development
                               Programming in Embedded C
Loop Control Instructions
Looping Instruction               Explanation
//skipping portion of a loop
while (expression)                Certain situation demands the skipping of a portion of a loop for some
{                                 conditions. The ‘continue’ statement used inside a loop will skip the rest of the
…………..;                           portion following it and will transfer the program control to the beginning of the
if (condition);                   loop.
continue;                         //for loop with skipping
………….;                            for (initialization; test for condition; update variable)
}                                 {
//do while with skipping          …………….;
do                                if (condition)
{                                 continue;
…………….;                           …………….;
if (condition)                    }
continue;
…………….;
}
while (expression);
                                                                                                                   26
Embedded Firmware Design & Development
  Programming in Embedded C - Loops
  //###########################################################
  //using while loop
  //###########################################################
   char *status_reg = (char *) 0x3000; //Declares memory mapped register
  //###########################################################
  //using do while loop
  //###########################################################
   char *status_reg = (char*) 0x3000;
  do
  {
  //###########################################################
  //using for loop
  //###########################################################
   char *status_reg = (char*) 0x3000;
  for (;(*status_reg!=0x01););
                                                                           27
       Embedded Firmware Design & Development
&arr[0] 0x8000
arr[0] 0x10
//Selective initialization
     arr[0] = 5;
     arr[1] = 10;
     arr[2] = 20;
     arr[3] = 3;
     arr[4] = 2;                                                                  29
          Embedded Firmware Design & Development
                                                 0x7E
                                                 0x7F
                                                                                      30
            Embedded Firmware Design & Development
input 0x45 10
 The content of memory location representing the variable input (0x45) can be
  accessed and modified by using a pointer of type same as the variable (char for
  the variable input in the example)
 In Embedded C the same is achieved through
                                                                                      31
            Embedded Firmware Design & Development
input 0x45 10
p 0x00 0x45
 Arrays are not equivalent to pointers and pointers are not equivalent to arrays
 The expression array name [] is equivalent to a pointer, of type specified by the
  array, to the first element of an array
 E.g. for the character array char arr[5], arr[ ] is equivalent to a character pointer
  pointing to the first element of array arr (This feature is referred as ‘equivalence
  of pointers and arrays’)
 The array features like accessing and modifying members of an array can be
  achieved using a pointer and pointer increment/decrement operators
 Arrays and pointer declarations are interchangeable when they are used as
  parameters to functions
                                                                                     33
            Embedded Firmware Design & Development
Programming in Embedded C – Characters & Strings
 Character is a one byte data type and it can hold values ranging from 0 to 255
   (unsigned character) or -128 to +127 (signed character)
 The term character literally refers to the alpha numeric characters (English
   alphabets A to Z (both small letters and Capital letters) and number
   representation from ‘0’ to ‘9’) and special characters like *, ?, ! Etc
 String is an array of characters
 A group of characters defined within a double quote represents a constant
   string
 ‘H’ is an example for a character, whereas “Hello” is an example for a string.
   String always terminates with a ‘\0’ character
 The ‘\0’ character indicates the string termination
 Whenever you declare a string using a character array, allocate space for the
   null terminator ‘\0’ in the array length
     E.g.   char name [ ] = “HELLO” ;
            char name [6] = {‘H’, ‘E’, ‘L’, ‘L’, ‘O’, ‘\0’};                  34
           Embedded Firmware Design & Development
 A function is a self contained and re-usable code snippet intended to perform a specific task
 ‘Embedded C’ supports two different types of functions namely
     library functions
     user defined functions
 Library functions are the built in functions which is either part of the standard ‘Embedded
  C’ library or user created library files
 printf(), scanf (), strcpy(), strcmp() etc are examples of standard library functions
 All library functions supported by a particular library is implemented and exported in the
  same
 A corresponding header (‘.h’) file for the library file provides information about the various
  functions available to user in a library file
 Users should include the header file corresponding to a particular library file for calling the
  functions from that library in the ‘C’ source file
 User defined functions are programmer created functions in the source file (Not in a library
  file) for various reasons like modularity, easy understanding of code, code reusability etc
                                                                                               38
             Embedded Firmware Design & Development
Programming in Embedded C – Functions
 Return type of a function tells – what is the data type of the value returning by the
  function on completion of its execution
 The general form of function declaration is
 The ‘Linkage Type’ specifies the linkage for the function. It can be either ‘external’
  or ‘internal’
 The ‘static’ keyword for the ‘Linkage Type’ specifies the linkage of the function as
  internal whereas the ‘extern’ ‘Linkage Type’ specifies ‘external’ linkage for the        39
  function
              Embedded Firmware Design & Development
Programming in Embedded C – Function Pointer
   structure is a variable holding a collection of data types (int, float, char, long etc)
   The data types can be either unique or distinct
   The tag ‘struct’ declares a variable as structure
   The general form of a structure declaration is
      struct      struct_name
      {
      //variable 1 declaration
      //variable 2 to declaration
      //……………
      //variable n declaration
      };
      struct employee
      {         char emp_name [20];// Allowed maximum length for name = 20
                int emp_code;
                char DOB [10]; // DD-MM-YYYY Format (10 character)
      };
                                                                                              41
           Embedded Firmware Design & Development
                     Programming in Embedded C – Structure Operations
 Structure variables are always stored in the memory of the target system with
  structure member variables in the same order as they are declared in the structure
  definition
 It is not necessary that the variables should be placed in continuous physical memory
  locations
 For multi byte processors (processors with word length greater than 1 byte (8 bits)), if
  the structure elements are arranged in memory in such a way that they can be
  accessed with lesser number of memory fetches, it definitely speeds up the operation
 structure padding is the process of arranging the structure elements in memory in a
  way facilitating increased execution speed
 As an example consider the following structure
             typedef struct
             {
                        char x;
                        int y;
             } exmpl;
                                                                                        43
             Embedded Firmware Design & Development
Let us analyze the various possibilities of storing the above structure within the memory.
            Memory
                          4x + 3         4x + 2         4x + 1           4x
            Address
                         Byte 2 of      Byte 1 of      Byte 0 of
              Data                                                     exmpl.x
                         exmpl.y        exmpl.y        exmpl.y
                                                                      Byte 3 of
              Data
                                                                      exmpl.y
            Memory
                        4(x + 1) + 3   4(x + 1) + 2    4(x + 1) + 1     4(x + 1)
            Address
                                                                                             44
          Embedded Firmware Design & Development
 Programming in Embedded C – Structure Padding (Packed Structure)
Memory
                4x + 3             4x + 2              4x + 1           4x
Address
Memory
             4(x + 1) + 3       4(x + 1) + 2          4(x + 1) + 1     4(x + 1)
Address
                                                                                  45
                 Embedded Firmware Design & Development
              Programming in Embedded C – Structure and Bit fields
 Bit field is a useful feature supported by structures for bit manipulation operation and
  setting up flags
 A set of bits in the structure bit field forms a ‘char’ or ‘int’ variable
 The general format of declaration of bit fields within structure is illustrated below
     struct    struct_name
     {
               data type (char or int) bit_var 1_name : bit_size,
               bit_var 2_name : bit_size,
               ………………………..,
               ………………………...,
               bit_var n_name : bit_size;
     };
As an illustrative example let us see how the PSW register of 8051 (which is a bit
addressable 8 bit register) can be represented in Embedded C
     struct PSW
               {
                             char             P:1,             /* Bit 0 of PSW : Parity Flag */
                                              :1,              /* Bit 1 of PSW : Unused */
                                              OV:1,            /* Bit 2 of PSW : Overflow Flag */
                                              RS0:1,           /* Bit 3 of PSW : Register Bank Select 0 bit */
                                              RS1 :1,          /* Bit 4 of PSW : Register Bank Select 1 bit */
                                              F0:1,            /* Bit 5 of PSW : User definable Flag */
                                              AC :1,           /* Bit 6 of PSW : Auxiliary Cary Flag */
                                              C:1;             /* Bit 7 of PSW : Carry Flag */
              /*Note that the operator ‘;’ is used after the last bit field to*/
              /* indicate end of bit field*/
};
 Each bit field variable is defined with a name and an associated bit size
  representation
 If some of the bits are unused in a packed fashion, the same can be skipped by                                 47
  merely giving the number of bytes to be skipped
             Embedded Firmware Design & Development
                          Programming in Embedded C – Union
 Union is a concept derived from structure and union declarations follow the same
  syntax as that of structures (structure tag is replaced by union tag)
 Though union looks similar to structure in declaration, it differs from structure in the
  memory allocation technique for the member variables
 Whenever a union variable is created, memory is allocated only to the member
  variable of union requiring the maximum storage size.
 For structure variables memory is allocated to each member variables
 Even if a union variable contains different member variables of different data types,
  existence is only for a single variable at a time
 The size of a union returns the storage size of its member variable occupying the
  maximum storage size
 The syntax for declaring union is given below
     union   union_name                           typedef union
      {                                             {
             //variable 1 declaration                       //variable 1 declaration
             //variable 2 to declaration                    //variable 2 to declaration
             //……………                                        //……………
             //variable n declaration                       //variable n declaration
       };                                           } union_name;                            48
               Embedded Firmware Design & Development
                              Programming in Embedded C – Union
Assuming the storage location required for ‘int’ as 4 bytes and for ‘char’ as 1 byte, the
memory allocated to the union variable ex1 will be as shown below
            Memory
                                                                                       ex1.z
            Address                 4x + 3            4x + 2              4x + 1        4x
/*Explicit declaration of character pointer pointing to 8 bit memory location, mapped at location 0x3007;
The ‘volatile’ keyword informs the cross compiler that the variable with ‘volatile’ qualifier is subject to
asynchronous change and there by the cross compiler turns off any optimization (assumptions on the
variable) for these variable
The general form of declaring a volatile variable is given below.
     //Declaring a volatile variable
     volatile data type variable name;
      //or
     data type volatile variable name;
     //Usage:
     //Declares volatile variable
     volatile char *status_reg = (char *) 0x3000;
     while (*status_reg!=0x01);               //Wait till status_reg = 0x01
                                                                                                         53
                    Embedded Firmware Design & Development
                       Programming in Embedded C – Volatile Qualifier
The ‘constant volatile’ Variable
Some variables used in embedded applications can be both ‘constant’ and ‘volatile’. A
‘Read only’ status register of a memory mapped device is a typical example for this. From
a user point of view the ‘Read only’ status registers can only be read but cannot modify.
Hence it is a constant variable. From the device point the contents can be modified at any
time by the device. So it is a volatile variable. Typical declarations are given below.
    volatile const int a;     // Constant volatile integer
    volatile const int* a;    //Pointer to a Constant volatile integer
Volatile pointer
Volatile pointers is subject to change at any point after they are initialized. Typical
examples are pointer to arrays or buffers modifiable by Interrupt Service Routines and
pointers in dynamic memory allocation. Pointers used in dynamic memory allocation can
be modified by the realloc() function. The general form of declaration of a volatile pointer
to a non-volatile variable is given below.
    data type* volatile variable name;
    E.g.       unsigned char* volatile a;
                                                                                          54
          Embedded Firmware Design & Development
                                                                         55
                     Embedded Firmware Design & Development
                           Programming in Embedded C – Infinite Loops
 Infinite loops are created using various loop control instructions like while (),
  do while (), for and goto labels
   //Infinite loop using while
   while (1)
                {
} while (1);
   for (; ; ;)
                 {
                 }
Bitwise OR
Operator ‘|’ performs Bitwise OR operations. Bitwise OR operation is usually performed
for selectively setting of bits and testing the current state of a bit (Bitwise ORing with ‘0’)
Bitwise NOT
Bitwise NOT operations negates (inverts) the state of a bit. The operator ‘~’ (tilde) is used
as the Bitwise NOT operator in C.
                                                                                                57
                     Embedded Firmware Design & Development
             Programming in Embedded C – Bit Manipulation Operations
Setting and Clearing and Bits
Bitwise OR operation sets a specified bit of a variable
Eg: flag = flag | 1; //Sets 0th Bit
Bitwise OR operation combined with left shift operation of ‘1’ is used for selectively
setting any bit in a variable. For example the following operation will set bit 6 of char
variable flag.
//Sets 6th bit of flag. Bit numbering starts with 0.
flag = flag | (1<<6);
//OR
flag |= (1<<6);                          //Equivalent to flag = flag | (1<<6);
Toggling Bits
Toggling a bit is performed to negate (toggle) the current state of a bit. If current state of a
specified bit is ‘1’, after toggling it becomes ‘0’ and vice versa. Toggling is also known as
inverting bits. The Bitwise XOR operator is used for toggling the state of a desired bit in an
operand.
flag ^= (1<<6);             //Toggle bit 6 of flag
                                                                                               58
               Embedded Firmware Design & Development
          Programming in Embedded C – Bit Manipulation Operations
Extracting /Inserting Bits from/to Packed Variables
Consider a 16 bit int variable with name ‘date’ which holds Date, Month and Year as
shown below
                         Bit 9 to 15                  Bit 5 to 8           Bit 0 to 4
              15                               9                     5                    0
                       Year (0 to 99)              Month (1 to 12)       Date (1 to 31)
Testing Bits
Bitwise operators can be used for checking the present status of a bit without modifying it for decision
making operations.
                                                                                                      59
             Embedded Firmware Design & Development
       Programming in Embedded C – Coding Interrupt Service Routines
 For Super loop based embedded firmware design, code the ISR as per the support from the
  cross-compiler in use for target processor.
 The example given below illustrates the coding of ISR in Embedded C for C51 Cross
  compiler for 8051 processor
            void interrupt_name (void)   interrupt x using y
            {
            /*Process Interrupt*/
            }
 interrupt_name is the function name for ISR
 The attribute ‘interrupt’ instructs the cross compiler that the associated function is an
  interrupt service routine
 The interrupt attribute takes an argument x which is an integer constant in the range 0 to 31
  (supporting 32 interrupts). This number is the interrupt number and it is essential for placing
  the generated hex code corresponding to the ISR in the corresponding Interrupt Vector
  Address
 using is an optional keyword for indicating which register bank is used for the general
  purpose Registers R0 to R7 (For more details on register banks and general purpose
  registers, refer to the hardware description of 8051)
 The argument y for using attribute can take values from 0 to 3 (corresponding to the register
  banks 0 to 3 of 8051)                                                                          60
            Embedded Firmware Design & Development
Programming in Embedded C – Reentrant Vs Recursive Functions
 A function which calls itself repeatedly is called a Recursive Function
 Using recursion, a complex problem is split into its single simplest form. The
  recursive function only knows how to solve that simplest case
 Recursive functions are useful in evaluating certain types of mathematical
  function, creating and accessing dynamic data structures such as linked lists or
  binary trees
 Functions which can be shared safely with several processes concurrently are
  called re-entrant functions
 When a re-entrant function is executing, another process can interrupt the
  execution and can execute the same re-entrant function
 The “another process” referred can be a thread in a multithreaded application
  or can be an Interrupt Service Routine (ISR)
 Re-entrant function is also referred as ‘pure’ function
 It is not necessary that all recursive functions are re-entrant. But a Re-entrant
  function can be invoked recursively by an application                           61
             Embedded Firmware Design & Development
Programming in Embedded C – Dynamic Memory Allocation
The conceptual view of storage of an application and the variables related to the application
in an Operating System based execution environment is represented as
           Dynamic Storage
             Memory
Alterable Data
                                                                                           62
               Embedded Firmware Design & Development
        Programming in Embedded C – Memory Management Functions
malloc()
malloc() function allocates a block of memory dynamically. The malloc() function reserves a block of
memory of size specified as parameter to the function, in the heap memory and returns a pointer of
type void.
            pointer = (pointer_type *) malloc (no. of bytes);
            Eg: ptr= (char *) malloc(50);
calloc()
The library function calloc() allocates multiple blocks of storage bytes and initializes each allocated
byte to zero.
            pointer = (pointer_type *) calloc (n, size of block);
             ptr= (char *) calloc (50,1);
free()
The ‘C’ memory management library function free() is used for releasing or de-allocating the memory
allocated in the heap memory by malloc() or calloc() functions
            free (ptr);
realloc()
realloc() function is used for changing the size of allocated bytes in a dynamically allocated memory
block
                                                                                                        63
            realloc (pointer, modified size);