File: CONTRIBUTING.rst

package info (click to toggle)
drgn 0.1.0-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 7,852 kB
  • sloc: python: 74,992; ansic: 54,589; awk: 423; makefile: 351; sh: 99
file content (301 lines) | stat: -rw-r--r-- 10,350 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
Contributing
============

Thanks for your interest in drgn! See below for how to build, test, code, and
submit changes for drgn.

Building
--------

The easiest way to develop drgn is by building and running it locally. Please
build with warnings enabled. Install the dependencies from the `installation
instructions <README.rst#from-source>`_, then run:

.. code-block:: console

    $ git clone https://github.com/osandov/drgn.git
    $ cd drgn
    $ CONFIGURE_FLAGS="--enable-compiler-warnings=error" python3 setup.py build_ext -i
    $ python3 -m drgn --help

Drgn can build, run, and pass its test suite on Python 3.8 or later. However,
many of the tools used as part of the development workflow do not support Python
versions once they have reached their end-of-life. Thus, your main drgn
development environment should use a Python version which is actively supported
upstream.

Testing
-------

Tests should be added for all features and bug fixes.

drgn's test suite can be run with:

.. code-block:: console

    $ python3 setup.py test

To run Linux kernel helper tests in a virtual machine on all supported kernels,
add ``-K``. See `vmtest <vmtest/README.rst>`_ for more details.

Tests can also be run manually with `unittest
<https://docs.python.org/3/library/unittest.html#command-line-interface>`_
after building locally:

.. code-block:: console

    $ python3 -m unittest discover -v

To run Linux kernel helper tests on the running kernel, this must be run as
root, and debug information for the running kernel must be available.

pre-commit
----------

Several linters and checks are run on every pull request. If you'd like to run
them locally prior to submission, you can install `pre-commit
<https://pre-commit.com/>`_:

.. code-block:: console

    $ pip install pre-commit

Then, you can either install the checks as Git hooks so that they're run when
creating a commit:

.. code-block:: console

    $ pre-commit install --install-hooks

Or you can run them manually:

.. code-block:: console

    $ pre-commit run --all-files

Coding Guidelines
-----------------

* Core functionality should be implemented in ``libdrgn`` and exposed to Python
  via the `C extension <libdrgn/python>`_. Only the CLI and helpers should be
  in pure Python.

C
^

drgn is written in GNU C11. C code in drgn mostly follows the `Linux kernel
coding style <https://www.kernel.org/doc/html/latest/process/coding-style.html>`_
with some slightly more modern preferences:

* Variables should be declared as close as possible to where they are used (as
  opposed to the C89 style of declaring everything at the top of a function).

  * As an exception, if a function has a local ``struct drgn_error *err``, it
    should usually be declared at the top of the function. (This is because
    must functions have such a variable, and it adds noise to have it in the
    middle of the function.)

* Scope guards and the `cleanup attribute
  <https://gcc.gnu.org/onlinedocs/gcc/Common-Variable-Attributes.html#index-cleanup-variable-attribute>`_
  should be used liberally.
* ``//``-style comments are preferred over ``/* */``.

  * As an exception, Doxygen comments should use ``/** */``.

  For example:

  .. code-block:: c

      /** Good example. */
      struct drgn_error *my_func(struct drgn_program *prog, size_t n)
      {
              struct drgn_error *err;

              _cleanup_free_ void *buf = malloc(n);
              if (!buf)
                      return &drgn_enomem;
              // 0xffff0000 is a nice address.
              err = drgn_program_read_memory(prog, buf, 0xffff0000, n, false);
              if (err)
                      return err;
              ...
              return NULL;
      }

  NOT:

  .. code-block:: c

      /* BAD example. */
      struct drgn_error *my_func(struct drgn_program *prog, size_t n) {
              struct drgn_error *err;
              void *buf;

              buf = malloc(n);
              if (!buf) {
                      return &drgn_enomem;
              }
              /* 0xffff0000 is a nice address. */
              err = drgn_program_read_memory(prog, buf, 0xffff0000, n, false);
              if (err) goto out;
              ...
              err = NULL;
      out:
              free(buf);
              return err;
      }

A few other guidelines/conventions:

* Constants should be defined as enums or ``static const`` variables rather
  than macros.
* Functions that can fail should return a ``struct drgn_error *`` (and return
  their result via an out parameter if necessary).
* Out parameters should be named ``ret`` (or suffixed with ``_ret`` if there
  are multiple) and be the last parameter(s) of the function.
* Functions that initialize an already allocated structure should be suffixed
  with ``_init`` and take the structure to initialize as the first argument,
  e.g., ``struct drgn_error *foo_init(struct foo *foo, int foo_flags)``.
* The matching function to deinitialize a structure should be suffixed with
  ``_deinit``, e.g., ``void foo_deinit(struct foo *foo)``. If possible, the
  definition should be placed directly after the definition of ``_init`` so
  that it is easier to visually verify that everything is cleaned up.
* Functions that allocate and initialize a structure should be suffixed with
  ``_create`` and either return the structure as an out parameter (e.g.,
  ``struct drgn_error *foo_create(int foo_flags, struct foo **ret)``) or as the
  return value if they can only fail with an out-of-memory error (e.g.,
  ``struct foo *foo_create(int foo_flags)``).
* The matching function to free an allocated structure should be suffixed with
  ``_destroy``, e.g., ``void foo_destroy(struct foo *foo)``. If possible, the
  definition should be placed directly after the definition of ``_create``.
  ``_destroy`` should usually allow a ``NULL`` argument, just like ``free()``.
* Functions that return a result in a ``struct drgn_object *`` parameter should
  only modify the object if the function succeeds.

drgn assumes some `implementation-defined behavior
<https://gcc.gnu.org/onlinedocs/gcc/C-Implementation.html>`_ for sanity:

* Signed integers are represented with two's complement.
* Bitwise operators on signed integers operate on the two's complement
  representation.
* Right shift of a signed integer type is arithmetic.
* Conversion to a signed integer type is modular.
* Casting between pointers and integers does not change the bit representation.

Python
^^^^^^

Python code in drgn should be compatible with Python 3.8 and newer.

Python code is formatted with `Black <https://github.com/psf/black>`_ and
`isort <https://github.com/PyCQA/isort>`_.

Type hints are required everywhere (including helpers and the C extension),
except in tests.

Public APIs should be documented with docstrings in reStructuredText format.

Non-public functions, classes, etc. should have names prefixed with an
underscore (``_``) and use normal comments instead of docstrings (this is
because drgn's `documentation generation system <docs/exts/drgndoc>`_ uses the
presence of a docstring to determine whether something should be included in
the Sphinx documentation).

Linux Kernel Helpers
^^^^^^^^^^^^^^^^^^^^

Linux kernel helpers should work on all `supported kernels
<https://drgn.readthedocs.io/en/latest/support_matrix.html#linux-kernel-versions>`_
if possible. This may require handling changes between kernel releases.

* Do NOT check the kernel version number to do this; Linux distributions often
  backport changes without updating the version number. Instead, use the
  presence or absence of variables, types, structure members, etc.
* Optimize for the latest kernel release, and follow "easier to ask for
  forgiveness than permission" (`EAFP
  <https://docs.python.org/3/glossary.html#term-EAFP>`_). For example, assume
  that a structure member from the latest release exists and catch the
  exception if it doesn't.
* Reference the diverging commit and version number in the format ``Linux
  kernel commit $abbreviated_commit_hash "$commit_subject" (in
  v$kernel_version)``.

  For example:

  .. code-block:: python3

      # Since Linux kernel commit 2f064a59a11f ("sched: Change
      # task_struct::state") (in v5.14), the task state is named "__state".
      # Before that, it is named "state".
      try:
          return task.__state
      except AttributeError:
          return task.state

  NOT:

  .. code-block:: python3

      # BAD
      if hasattr(task, "state"):
          return task.state
      else:
          return task.__state

* Document the expected C types of arguments and return values. For example:

  .. code-block:: python3

      def cgroup_parent(cgrp: Object) -> Object:
          """
          Return the parent cgroup of the given cgroup if it exists, ``NULL``
          otherwise.

          :param cgrp: ``struct cgroup *``
          :return: ``struct cgroup *``
          """
          ...

Submitting PRs
--------------

Pull requests and issues are always welcome. Feel free to start a discussion
with a prototype.

Signing Off
^^^^^^^^^^^

All commits must be signed off (i.e., ``Signed-off-by: Jane Doe
<janedoe@example.org>``) as per the `Developer Certificate of Origin
<https://developercertificate.org/>`_. ``git commit -s`` can do this for you.

Separating Changes
^^^^^^^^^^^^^^^^^^

Each logical change should be a separate commit. For example, if a PR adds new
functionality to the core library and a new helper that uses the new
functionality, the core change and the helper should be separate commits. This
makes code review much easier.

Each commit should build, pass tests, follow coding guidelines, and run
correctly. (In other words, within a PR, later commits often build on top of
earlier commits, but later commits shouldn't need to "fix" earlier commits.)
This makes it easier to track down problems with tools like ``git bisect``
which may check out any commit in the middle of a PR.

Commit Messages
^^^^^^^^^^^^^^^

The template for a good commit message is:

.. code-block:: none

    One line summary

    Longer explanation including more details, background, and/or
    motivation.

    Signed-off-by: Jane Doe <janedoe@example.org>

See `this post <https://chris.beams.io/posts/git-commit/>`_ for more
information about writing good commit messages.