How to write documentation
Gramine uses Sphinx to generate the documentation in HTML form. The
documentation is hosted on Read The Docs at https://gramine.readthedocs.io/.
Documentation is generally written as reStructuredText files which are placed
in Documentation/
directory. See Sphinx’ reST primer for short
introduction into syntax.
For code written in C, we use Doxygen and Breathe, which is
a Sphinx’ plugin for including Doxygen documentation. Documentation of
C language API should be written as Doxygen comments (see Doxygen manual)
and then included in one of the .rst
files (with appropriate description)
using one of the Breathe directives, like .. doxygenfunction::
or ..
doxygenstruct::
.
The documentation should be written with html
builder of Sphinx in mind. The
manpages/
subdirectory also targets manpage
builder. Other builders
(like latex
) may be considered in the future, but for now their output is
not published.
Note
A note about terminology:
html
, latex
and manpage
, and also others, are Sphinx “builders”:
https://www.sphinx-doc.org/en/master/man/sphinx-build.html#cmdoption-sphinx-build-b.
Sphinx can output many different formats, some of them have overlapping usage
(both html
and latex
usually output full handbook, the difference is
screen vs print), some are specialized (manpage
processes only selected
documents for man; those documents may or may not be also used by other
builders).
When launched through make
(like make -C Documentation html
), this
becomes “target” in Make terminology.
Installing prerequisites
Change directory to Documentation
, install prerequisites including Doxygen,
Breathe and Sphinx. Gramine documentation requires version 3.4 of Sphinx.
# change directory to Documentation
cd Documentation
# install prerequisites
sudo apt-get install doxygen
python3 -m pip install -r requirements.txt
Building documentation
Execute make
command in Documentation
directory, specifying the
appropriate target. The output is in the _build
directory.
# build targets "html" and "man"
make html man
# example: view html output
firefox _build/html/index.html
# example: view man output
man _build/man/gramine-manifest.1
Preferred reST style
(This is adapted from Python’s style guide).
Use 3-space tab in
.rst
files to align the indentation with reST explicit markup, which begins with two dots and a space.Wrap the paragraphs at 80th character. But don’t wrap verbatim text like logs and use applicable style when wrapping code examples (see Coding style guidelines).
For headers, use Python convention for header hierarchy:
#
with overline,*
with overline,=
,-
,^
,"
.
Example:
################################### Very top level header (in TOC etc.) ################################### ******************* Less than top level ******************* Per-file header =============== Section header -------------- Subsection header ^^^^^^^^^^^^^^^^^ Subsubsection header """"""""""""""""""""
This means most documents use only
=
and-
adornments.Tip
- For vim users:
you can enter the
-
underlines using the key combinationyypVr-
and the other adornments with similar combinations.- For Emacs users:
Read more at https://docutils.sourceforge.io/docs/user/emacs.html.
Use
|~|
to insert non-breaking space. This should be added after one-letter words and where otherwise appropriate:This is a |~| function.
This substitution is added to all documents processed by Sphinx. Don’t use this for files processed also by other software (like
README.rst
, which is both rendered by GitHub and included inindex.rst
) - GitHub renders those spaces incorrectly.
Documentation of the code should be organized into files by logical concepts, as they fit into programmer’s mind. Ideally, this should match the source files, if those files were organized correctly in the first place, but the reality may be different. In case of doubt, place them as they fit the narrative of the document, not as they are placed in the source files.
Documents should be grouped by general areas and presented using
.. toctree::
directive in index.rst
file. This causes them to be
included in TOC in the main document and also in sidebar on RTD.
Preferred Doxygen style
Prefer Qt-style
/*!
and\param
:/*! * \brief Sum two integers. * * \param first First addend. * \param second Second addend. * * \returns Sum of the arguments. Sometimes a longer description is needed, then it should be * wrapped and aligned like this. * * Longer description goes here (only for functions complex enough to deserve it). */ int foo(int first, int second) { return first + second; }
There is a |~| very special function :c:func:`foo`: .. doxygenfunction:: foo It's an example function, but is documented!
In reST, do not use
autodoxygen
directives, and especially do not use.. doxygenfile::
, because documentation should be written as prose, not a coredump. Write an explanation, how the things go together and place the.. doxygenfunction::
directives where aproppriate.You can use
\rst
and\endrst
to write reST in Doxygen comments:/*! * \brief An example function * * \rst * .. note:: * * This works! * \endrst */
Further reading
The Hitchhiker’s Guide to Documentation divided by audience (role in the project), with references to good real-world examples