Markdown files are used to write most documentation. The main Markdown tool is Hugo.
The Markdown processing is done using the build_docs.py
tool in the util
directory.
As with all style guides the intention is to:
Unless otherwise noted, the following terminology conventions apply to this style guide:
Justify exceptions with a comment.
No style guide is perfect. There are times when the best path to a working design, or for working around a tool issue, is to simply cut the Gordian Knot and create code that is at variance with this style guide. It is always okay to deviate from the style guide by necessity, as long as that necessity is clearly justified by a brief comment.
In OpenTitan, most--but not all--Markdown documents will be rendered to HTML before they are presented to the reader. However, README files are an important exception, and so the recommended line-wrapping style differs for these two types of files.
Rendered Files: Files which are intended to be rendered before viewing should have exactly one sentence per line, with no line breaks in the middle of a sentence. This way change reviews will highlight only those sentences which are modified. Though the long line lengths make the files slightly less convenient to read from the command-line, this greatly simplifies the review process. When reviewing Markdown changes, every altered sentence will be included in its entirety in the file diff.
README Files: README files should wrap lines at under 80 characters. This ensures that the source is readable without any Markdown processing. Please note, however, that re-wrapping a paragraph after an insertion or deletion tends to cause longer diffs when the change is reviewed. When making changes to a document using this style, please consider allowing short lines rather than a full re-wrap after minor edits. Then occasionally separate commits can be used that only do re-wrapping of the paragraphs.
The title of the document should be provided using the title
field in the frontmatter.
Headings and sections are given ID tags to allow cross references. The ID is the text of the heading, converted to lower case and with spaces converted to -
. Thus ### Headings and sections
gets the ID headings-and-sections
and can be referenced using the Markdown hyperlink syntax [link text](#headings-and-sections)
.
Headings and sections are added to the table of contents.
Pictures can be included using the standard Markdown syntax (![Alt Text](url)
). The preferred format is Scalable Vector Graphics (.svg
), alternatively Portable Network Graphics (.png
).
Waveforms can be included by adding wavejson code surrounded by {{</* wavejson */>}}
shortcode tags.
Where possible, please restrict Markdown text to the ASCII character set to avoid downstream tool issues. Unicode may be used when referring to proper names.
Comments are rare, but should be used where needed. Use the HTML <!--
and -->
as the comment delimiters.
The Markdown files should use the .md
file extension.
Typically the Markdown file for an IP block follows the same outline.
The header instantiates the standard document header and reads the Hjson description of the module.
--- title: "Example IP Block" ---
This is followed by some boiler-plate comments.
This document specifies Name hardware IP functionality. This module conforms to the [Comportable guideline for peripheral functionality.]({{</* relref "doc/rm/comportability_specification" */>}}) See that document for integration overview within the broader top level system.
The next section summarizes the feature set of the IP block.
## Features * Bulleted list * Of main features
There then follows a general description of the IP
## Description Description of the IP.
The Compatibility information will allow device driver writers to identify existing code that can be used directly or with minor changes.
This section is primarily of interest to software engineers.
## Compatability Notes on if the IP register interface is compatible with any existing register interface. Also note any differences. For example: Matches 16550 UART interface but registers are at 32-bit word offsets.
The next major section is a more detailed operational description of the module.
# Theory of Operations
Conventionally one of the first sections includes a block diagram and a description.
Should be useful to hardware designers, verification engineers and software engineers.
## Block Diagram ![Name Block Diagram](block_diagram.svg)
There should be a section containing the automatically generated description of the IP including the signals, interrupts and alerts that it uses.
Primary user is the SoC integrator, but useful for everyone.
Note that the interrupt descriptions are also automatically placed in the interrupt status register bit descriptions, which is the most likely place for software engineers to reference them.
## Hardware Interfaces
The organization of the design details section is done to suit the module.
## Design Details Details of the design. ### Many third level headings
There are probably waveforms embedded here:
{{</* wavejson */>}} { signal: [ { name: 'Clock', wave: 'p............' }, ] } {{</* /wavejson */>}}
The final major section is the software user guide and describes using the IP and notes on writing device drivers. Code fragments are encouraged.
This section is primarily for software engineers, but it is expected that the code fragments are used by verification engineers.
# Programmers Guide
One important thing here is to show the order of initialization that has been tested in the verification environment. In most cases other orders will work, and may be needed by the software environment, but it can be helpful in tracking down bugs for the validated sequence to be described!
## Initialization
if (...) { a = ... }
Other sections cover different use cases and example code fragments.
## Use case A (eg Transmission) ## Use case B (eg Reception)
It is important to include a discussion of error conditions.
## Error conditions
Also comment on anything special about interrupts, potentially including the priority order for servicing interrupts.
## Interrupt Handling
The document should end with the automatically generated register tables.
## Register Table {{</* registers "hw/ip/component/data/component.hjson" */>}}
To allow cut/paste of the default structure, here is an uncommented version:
--- title: Name HWIP Technical Specification --- # Overview This document specifies Name hardware IP functionality. This module conforms to the [Comportable guideline for peripheral functionality.]({{< relref "doc/rm/comportability_specification" >}}) See that document for integration overview within the broader top level system. ## Features * Bulleted list ## Description ## Compatibility # Theory of Operations ## Block Diagram ![Name Block Diagram](block_diagram.svg) ## Hardware Interfaces {{</* incGenFromIpDesc "../data/component.hjson" "hwcfg" */>}} ## Design Details ### Many third level headings # Programmers Guide ## Initialization ## Use case A (eg Transmission) ## Use case B (eg Reception) ## Error conditions ## Interrupt Handling ## Register Table {{</* incGenFromIpDesc "../data/component.hjson" "registers" */>}}