| # The OpenTitan DIF Library |
| |
| A DIF is a "Device Interface Function". DIFs are low-level routines for |
| accessing the hardware functionality directly, and are agnostic to the |
| particular environment or context they are called from. The intention is that |
| DIFs can be used during design verification, and during early silicon |
| verification, and by the high-level driver software in production firmware. |
| |
| This subtree provides headers and libraries known collectively as the DIF |
| libraries. |
| |
| There is one DIF library per hardware IP, and each one contains the DIFs |
| required to actuate all of the specification-required functionality of the |
| hardware they are written for. |
| |
| ## DIF Style Guide |
| |
| DIFs are very low-level software, so they have a more rigorous coding style than |
| other parts of the codebase. |
| |
| DIFs should follow the [OpenTitan C/C++ style |
| guide](https://docs.opentitan.org/doc/rm/c_cpp_coding_style/) where it does not |
| contradict with the guidelines below. |
| |
| The guidelines below apply to writing DIFs, and code should be written in a |
| similar style to the existing DIF libraries in this directory. |
| |
| ### DIF Library Guidance |
| |
| * DIF libraries must be written in C. |
| * DIF libraries can only depend on the following headers (and their associated |
| libraries) from the `sw/device/lib/base` directory: |
| * `sw/device/lib/base/mmio.h` |
| * `sw/device/lib/base/memory.h` |
| * DIF libraries must not depend on other DIF libraries. Exercising DIF |
| functionality may require an environment set up using another DIF library, but |
| DIFs must not call DIFs in other DIF libraries. |
| |
| ### DIF Guidance |
| |
| The following rules must be followed by public DIF functions (those declared in |
| the DIF library's header file). Internal DIF functions (those declared `static` |
| and not declared in the DIF library's header file) should follow these rules but |
| there are some relaxations of these rules for them described at the end. |
| |
| * DIF declarations must match their definitions exactly. |
| * Scalar arguments must not be declared `const` or `volatile` (cv-qualified) |
| in DIF signatures. |
| |
| * DIFs must use enum return codes rather than booleans for reporting errors. If |
| a DIF can either error or instead produce a value, it should return an enum |
| return code, and use an out-parameter for returning the produced value. |
| * DIFs that cannot error and that do not return a value should return `void`. |
| |
| * DIFs must check their arguments against preconditions using "guard |
| statements". A guard statement is a simple if statement at the start of a |
| function which only returns an error code if the preconditions are not met. |
| Guard statements must cover the following checks: |
| * DIFs must ensure their pointer arguments are non-null, unless that pointer |
| is for an optional out-parameter. Arguments typed `mmio_region_t` are not |
| pointers, and cannot meaningfully be checked for non-nullness. |
| * DIFs must ensure, if they only accept a subset of an enum, that the argument |
| is within that subset. However, DIFs should assume, for checking |
| preconditions, that any enum argument is one of the enum variants. |
| * DIFs must not have side-effects before any guard statements. Side-effects |
| include (but are not limited to) writing to memory, including memory-mapped |
| hardware, and modifying processor CSRs. |
| |
| * Switch statements in DIFs must always have a default case, including when |
| switching on an enum value (an "enum switch"). |
| * The default case of an enum switch must report an error for values that are |
| not an enum variant. |
| * Enum switches do not need a `case` for enum variants that are unreachable |
| due to a guard statement. |
| |
| * DIFs must use `sw/device/lib/base/mmio.h` for accessing memory-mapped |
| hardware. DIFs must not use `sw/device/lib/base/memory.h` for accessing |
| memory-mapped hardware. |
| |
| * Internal DIF functions, which are not intended to be part of a public DIF |
| library interface, must not be declared in the DIF library header, and must be |
| marked `static`. |
| * `static` DIF functions should not be marked `static inline`. |
| * An internal DIF function does not need to check preconditions, if all the |
| DIF functions that call it have already checked that precondition. |