187 lines
9.4 KiB
ReStructuredText
187 lines
9.4 KiB
ReStructuredText
.. SPDX-License-Identifier: GPL-2.0
|
|
|
|
bcachefs coding style
|
|
=====================
|
|
|
|
Good development is like gardening, and codebases are our gardens. Tend to them
|
|
every day; look for little things that are out of place or in need of tidying.
|
|
A little weeding here and there goes a long way; don't wait until things have
|
|
spiraled out of control.
|
|
|
|
Things don't always have to be perfect - nitpicking often does more harm than
|
|
good. But appreciate beauty when you see it - and let people know.
|
|
|
|
The code that you are afraid to touch is the code most in need of refactoring.
|
|
|
|
A little organizing here and there goes a long way.
|
|
|
|
Put real thought into how you organize things.
|
|
|
|
Good code is readable code, where the structure is simple and leaves nowhere
|
|
for bugs to hide.
|
|
|
|
Assertions are one of our most important tools for writing reliable code. If in
|
|
the course of writing a patchset you encounter a condition that shouldn't
|
|
happen (and will have unpredictable or undefined behaviour if it does), or
|
|
you're not sure if it can happen and not sure how to handle it yet - make it a
|
|
BUG_ON(). Don't leave undefined or unspecified behavior lurking in the codebase.
|
|
|
|
By the time you finish the patchset, you should understand better which
|
|
assertions need to be handled and turned into checks with error paths, and
|
|
which should be logically impossible. Leave the BUG_ON()s in for the ones which
|
|
are logically impossible. (Or, make them debug mode assertions if they're
|
|
expensive - but don't turn everything into a debug mode assertion, so that
|
|
we're not stuck debugging undefined behaviour should it turn out that you were
|
|
wrong).
|
|
|
|
Assertions are documentation that can't go out of date. Good assertions are
|
|
wonderful.
|
|
|
|
Good assertions drastically and dramatically reduce the amount of testing
|
|
required to shake out bugs.
|
|
|
|
Good assertions are based on state, not logic. To write good assertions, you
|
|
have to think about what the invariants on your state are.
|
|
|
|
Good invariants and assertions will hold everywhere in your codebase. This
|
|
means that you can run them in only a few places in the checked in version, but
|
|
should you need to debug something that caused the assertion to fail, you can
|
|
quickly shotgun them everywhere to find the codepath that broke the invariant.
|
|
|
|
A good assertion checks something that the compiler could check for us, and
|
|
elide - if we were working in a language with embedded correctness proofs that
|
|
the compiler could check. This is something that exists today, but it'll likely
|
|
still be a few decades before it comes to systems programming languages. But we
|
|
can still incorporate that kind of thinking into our code and document the
|
|
invariants with runtime checks - much like the way people working in
|
|
dynamically typed languages may add type annotations, gradually making their
|
|
code statically typed.
|
|
|
|
Looking for ways to make your assertions simpler - and higher level - will
|
|
often nudge you towards making the entire system simpler and more robust.
|
|
|
|
Good code is code where you can poke around and see what it's doing -
|
|
introspection. We can't debug anything if we can't see what's going on.
|
|
|
|
Whenever we're debugging, and the solution isn't immediately obvious, if the
|
|
issue is that we don't know where the issue is because we can't see what's
|
|
going on - fix that first.
|
|
|
|
We have the tools to make anything visible at runtime, efficiently - RCU and
|
|
percpu data structures among them. Don't let things stay hidden.
|
|
|
|
The most important tool for introspection is the humble pretty printer - in
|
|
bcachefs, this means `*_to_text()` functions, which output to printbufs.
|
|
|
|
Pretty printers are wonderful, because they compose and you can use them
|
|
everywhere. Having functions to print whatever object you're working with will
|
|
make your error messages much easier to write (therefore they will actually
|
|
exist) and much more informative. And they can be used from sysfs/debugfs, as
|
|
well as tracepoints.
|
|
|
|
Runtime info and debugging tools should come with clear descriptions and
|
|
labels, and good structure - we don't want files with a list of bare integers,
|
|
like in procfs. Part of the job of the debugging tools is to educate users and
|
|
new developers as to how the system works.
|
|
|
|
Error messages should, whenever possible, tell you everything you need to debug
|
|
the issue. It's worth putting effort into them.
|
|
|
|
Tracepoints shouldn't be the first thing you reach for. They're an important
|
|
tool, but always look for more immediate ways to make things visible. When we
|
|
have to rely on tracing, we have to know which tracepoints we're looking for,
|
|
and then we have to run the troublesome workload, and then we have to sift
|
|
through logs. This is a lot of steps to go through when a user is hitting
|
|
something, and if it's intermittent it may not even be possible.
|
|
|
|
The humble counter is an incredibly useful tool. They're cheap and simple to
|
|
use, and many complicated internal operations with lots of things that can
|
|
behave weirdly (anything involving memory reclaim, for example) become
|
|
shockingly easy to debug once you have counters on every distinct codepath.
|
|
|
|
Persistent counters are even better.
|
|
|
|
When debugging, try to get the most out of every bug you come across; don't
|
|
rush to fix the initial issue. Look for things that will make related bugs
|
|
easier the next time around - introspection, new assertions, better error
|
|
messages, new debug tools, and do those first. Look for ways to make the system
|
|
better behaved; often one bug will uncover several other bugs through
|
|
downstream effects.
|
|
|
|
Fix all that first, and then the original bug last - even if that means keeping
|
|
a user waiting. They'll thank you in the long run, and when they understand
|
|
what you're doing you'll be amazed at how patient they're happy to be. Users
|
|
like to help - otherwise they wouldn't be reporting the bug in the first place.
|
|
|
|
Talk to your users. Don't isolate yourself.
|
|
|
|
Users notice all sorts of interesting things, and by just talking to them and
|
|
interacting with them you can benefit from their experience.
|
|
|
|
Spend time doing support and helpdesk stuff. Don't just write code - code isn't
|
|
finished until it's being used trouble free.
|
|
|
|
This will also motivate you to make your debugging tools as good as possible,
|
|
and perhaps even your documentation, too. Like anything else in life, the more
|
|
time you spend at it the better you'll get, and you the developer are the
|
|
person most able to improve the tools to make debugging quick and easy.
|
|
|
|
Be wary of how you take on and commit to big projects. Don't let development
|
|
become product-manager focused. Often time an idea is a good one but needs to
|
|
wait for its proper time - but you won't know if it's the proper time for an
|
|
idea until you start writing code.
|
|
|
|
Expect to throw a lot of things away, or leave them half finished for later.
|
|
Nobody writes all perfect code that all gets shipped, and you'll be much more
|
|
productive in the long run if you notice this early and shift to something
|
|
else. The experience gained and lessons learned will be valuable for all the
|
|
other work you do.
|
|
|
|
But don't be afraid to tackle projects that require significant rework of
|
|
existing code. Sometimes these can be the best projects, because they can lead
|
|
us to make existing code more general, more flexible, more multipurpose and
|
|
perhaps more robust. Just don't hesitate to abandon the idea if it looks like
|
|
it's going to make a mess of things.
|
|
|
|
Complicated features can often be done as a series of refactorings, with the
|
|
final change that actually implements the feature as a quite small patch at the
|
|
end. It's wonderful when this happens, especially when those refactorings are
|
|
things that improve the codebase in their own right. When that happens there's
|
|
much less risk of wasted effort if the feature you were going for doesn't work
|
|
out.
|
|
|
|
Always strive to work incrementally. Always strive to turn the big projects
|
|
into little bite sized projects that can prove their own merits.
|
|
|
|
Instead of always tackling those big projects, look for little things that
|
|
will be useful, and make the big projects easier.
|
|
|
|
The question of what's likely to be useful is where junior developers most
|
|
often go astray - doing something because it seems like it'll be useful often
|
|
leads to overengineering. Knowing what's useful comes from many years of
|
|
experience, or talking with people who have that experience - or from simply
|
|
reading lots of code and looking for common patterns and issues. Don't be
|
|
afraid to throw things away and do something simpler.
|
|
|
|
Talk about your ideas with your fellow developers; often times the best things
|
|
come from relaxed conversations where people aren't afraid to say "what if?".
|
|
|
|
Don't neglect your tools.
|
|
|
|
The most important tools (besides the compiler and our text editor) are the
|
|
tools we use for testing. The shortest possible edit/test/debug cycle is
|
|
essential for working productively. We learn, gain experience, and discover the
|
|
errors in our thinking by running our code and seeing what happens. If your
|
|
time is being wasted because your tools are bad or too slow - don't accept it,
|
|
fix it.
|
|
|
|
Put effort into your documentation, commmit messages, and code comments - but
|
|
don't go overboard. A good commit message is wonderful - but if the information
|
|
was important enough to go in a commit message, ask yourself if it would be
|
|
even better as a code comment.
|
|
|
|
A good code comment is wonderful, but even better is the comment that didn't
|
|
need to exist because the code was so straightforward as to be obvious;
|
|
organized into small clean and tidy modules, with clear and descriptive names
|
|
for functions and variable, where every line of code has a clear purpose.
|