Difference between revisions of "CodingStandard"

From GnuCash
Jump to: navigation, search
(Add C++ section.)
(C++ Format: comment style with a bit of discussion)
(4 intermediate revisions by 3 users not shown)
Line 1: Line 1:
 +
This page describes the standard coding style throughout the gnucash source code.
 +
 
= C Format =
 
= C Format =
 
Gnucash is fairly relaxed about code format, but we will periodically run a reformatter called [http://astyle.sourceforge.net/astyle.html Artistic Style] (astyle) over the code to clean it up. To reduce the need for this (it messes up VCS history, particularly "blame"), please format your code as follows.
 
Gnucash is fairly relaxed about code format, but we will periodically run a reformatter called [http://astyle.sourceforge.net/astyle.html Artistic Style] (astyle) over the code to clean it up. To reduce the need for this (it messes up VCS history, particularly "blame"), please format your code as follows.
  
 
In general, follow the [http://www.gnu.org/prep/standards/standards.html#Formatting GNU format] '''except''' use four spaces instead of two for the indents, and ''don't'' indent the braces. To summarize, a properly formatted function will look like this:
 
In general, follow the [http://www.gnu.org/prep/standards/standards.html#Formatting GNU format] '''except''' use four spaces instead of two for the indents, and ''don't'' indent the braces. To summarize, a properly formatted function will look like this:
 
+
<syntaxhighlight lang="C">
 
   guint
 
   guint
 
   gnc_account_foo (Account *account, gpointer bar)
 
   gnc_account_foo (Account *account, gpointer bar)
Line 15: Line 17:
 
       return salt;
 
       return salt;
 
   }
 
   }
 
+
</syntaxhighlight>
 
Please keep lines under 80 characters. If you need to wrap, line up the function arguments like this:
 
Please keep lines under 80 characters. If you need to wrap, line up the function arguments like this:
 
+
<syntaxhighlight lang="C">
 
   gnc_account_function_with_a_lot_of_paramters (LongTypeName foo, LongerTypeName *bar,
 
   gnc_account_function_with_a_lot_of_paramters (LongTypeName foo, LongerTypeName *bar,
 
                                                 TypeName baz)
 
                                                 TypeName baz)
 
   {
 
   {
 
   }
 
   }
 +
</syntaxhighlight>
 
We don't do the wide separation of names and aligned parameters, so ''don't'' do this:
 
We don't do the wide separation of names and aligned parameters, so ''don't'' do this:
 +
<syntaxhighlight lang="C">
 
   void gnc_account_foo              (Account    *bar,
 
   void gnc_account_foo              (Account    *bar,
 
                                       Split      *baz,
 
                                       Split      *baz,
Line 28: Line 32:
 
   Split *gnc_account_pepper          (Account    *salt,
 
   Split *gnc_account_pepper          (Account    *salt,
 
                                       Transaction *sausage);
 
                                       Transaction *sausage);
 
+
</syntaxhighlight>
 
''Instead, do it this way:''
 
''Instead, do it this way:''
 +
<syntaxhighlight lang="C">
 
   void gnc_account_foo (Account *bar, Split *baz, gpointer waldo);
 
   void gnc_account_foo (Account *bar, Split *baz, gpointer waldo);
 
   Split *gnc_account_pepper (Account *salt, Transaction *sausage);
 
   Split *gnc_account_pepper (Account *salt, Transaction *sausage);
 
+
</syntaxhighlight>
 
For the record, the astyle commandline we use (with astyle version 1.24; other versions might result in slightly different formatting) is
 
For the record, the astyle commandline we use (with astyle version 1.24; other versions might result in slightly different formatting) is
 
   astyle --indent=spaces=4 --brackets=break --pad-oper  *.[hc]
 
   astyle --indent=spaces=4 --brackets=break --pad-oper  *.[hc]
Line 38: Line 43:
  
 
= C++ Format =
 
= C++ Format =
C++ follows the C format guidelines above, with the following modifications:
+
[[C++]] follows the C format guidelines above, with the following modifications:
* Namespaces shall be all lower case, and the outermost one shall be named '''gnucash'''.
+
* Namespaces shall be all lower case, and the outermost one shall be named '''gnc'''.
* All '''identifiers''' shall be '''camel-cased'''.
+
* Identifier formats:
** '''Typenames''' (i.e. class, struct, enum, and POD type-aliases) shall begin with an upper-case letter.
+
** '''Typenames''' (i.e. class, struct, enum, and POD type-aliases) shall be camel-cased.
** '''Function''' and '''object''' (including enumerator) names shall begin with a lower-case letter.
+
** '''Function''' and '''object''' names shall be lower-case and shall have words separated by underscores.
 +
** '''Preprocessor Macros''' and '''enumeration names''' shall be upper-case with words separated by underscores.
  
 
In addition, certain types of variables shall have prefixes denoting their roles:
 
In addition, certain types of variables shall have prefixes denoting their roles:
Line 50: Line 56:
 
* '''Free static variables''': '''s_'''. Free static variables means statics which are not local and not members of a class or struct, regardless of scope.
 
* '''Free static variables''': '''s_'''. Free static variables means statics which are not local and not members of a class or struct, regardless of scope.
  
Headers and implementation files should be named for the class they declare or implement. In general any file should declare or implement only one class.
+
Other Formatting:
 +
* Single-line standalone comments can use either C++ style or C style. End-of-line comments should always use C++ style. The preferred style for multi-line comments is currently under discussion.
 +
* Headers and implementation files should be named for the class they declare or implement. In general any file should declare or implement only one class.
  
 
== Coding Guidance ==
 
== Coding Guidance ==
  
* Write modern, idiomatic C++ using the new features of C++11 and later.
+
* Write modern, idiomatic C++ using the new features of C++11; use the -std= CXXFLAGS value in configure.ac to see if you can use features introduced in C++14 or C++17 (as of this writing you may not). In particular:
* Maximize use of the C++ standard library, especially the algorithms, and Boost.
+
** Prefer curly braces for initializers and <tt>auto</tt> for variable declarations.
* Keep thread-safety and reentrancy in mind: Modern processors are mostly multi-core, and modern operating systems and dispatchers will multi-thread programs on their own if possible.
+
** Declare and initialize variables in one statement at the point that they're first used.
* Avoid adding dependencies if possible. If you must, ensure that their licenses are compatible with GnuCash's and that we comply with any license requirements like crediting the copyright holder in documentation. We may not distribute dependencies on Linux and BSD, but we do on Windows and MacOSX.
+
** Use templates instead of copying code and especially instead of preprocessor macros.
 +
** Prefer passing references to pointers.
 +
** '''Do not ''ever''''' pass void* (aka gpointer) in C++ code. Use templates or class hierarchies to enforce type safety.
  
 
= Framework =
 
= Framework =
Line 65: Line 75:
 
= Guile and Scheme =
 
= Guile and Scheme =
 
Gnucash is partly implemented in a [http://groups.csail.mit.edu/mac/projects/scheme/ Scheme] dialect called [http://www.gnu.org/s/guile/ Guile]. (It was originally written mostly in Guile, but that implementation was largely replaced with C several years ago.) In particular, the reports system and part of the business system are written in Guile. To support that, most of the core "engine" API is wrapped and accessible from Guile.
 
Gnucash is partly implemented in a [http://groups.csail.mit.edu/mac/projects/scheme/ Scheme] dialect called [http://www.gnu.org/s/guile/ Guile]. (It was originally written mostly in Guile, but that implementation was largely replaced with C several years ago.) In particular, the reports system and part of the business system are written in Guile. To support that, most of the core "engine" API is wrapped and accessible from Guile.
 +
 +
= Graphical User Interface =
 +
See [[GUI Guidelines‎‎]].

Revision as of 20:53, 23 June 2018

This page describes the standard coding style throughout the gnucash source code.

C Format

Gnucash is fairly relaxed about code format, but we will periodically run a reformatter called Artistic Style (astyle) over the code to clean it up. To reduce the need for this (it messes up VCS history, particularly "blame"), please format your code as follows.

In general, follow the GNU format except use four spaces instead of two for the indents, and don't indent the braces. To summarize, a properly formatted function will look like this:

  guint
  gnc_account_foo (Account *account, gpointer bar)
  {
       Split *baz;
       guint salt;
       if (gnc_split_waldo (baz) > 0)
       {
            salt = gnc_split_pepper (baz);
       }
       return salt;
  }

Please keep lines under 80 characters. If you need to wrap, line up the function arguments like this:

  gnc_account_function_with_a_lot_of_paramters (LongTypeName foo, LongerTypeName *bar,
                                                TypeName baz)
  {
  }

We don't do the wide separation of names and aligned parameters, so don't do this:

  void gnc_account_foo               (Account     *bar,
                                      Split       *baz,
                                      gpointer     waldo);
  Split *gnc_account_pepper          (Account     *salt,
                                      Transaction *sausage);

Instead, do it this way:

  void gnc_account_foo (Account *bar, Split *baz, gpointer waldo);
  Split *gnc_account_pepper (Account *salt, Transaction *sausage);

For the record, the astyle commandline we use (with astyle version 1.24; other versions might result in slightly different formatting) is

 astyle --indent=spaces=4 --brackets=break --pad-oper  *.[hc]

The rationale for the arguments is contained in this email.

C++ Format

C++ follows the C format guidelines above, with the following modifications:

  • Namespaces shall be all lower case, and the outermost one shall be named gnc.
  • Identifier formats:
    • Typenames (i.e. class, struct, enum, and POD type-aliases) shall be camel-cased.
    • Function and object names shall be lower-case and shall have words separated by underscores.
    • Preprocessor Macros and enumeration names shall be upper-case with words separated by underscores.

In addition, certain types of variables shall have prefixes denoting their roles:

  • Member variables: m_.
  • Static member variables (a.k.a class variables): c_.
  • Static constants: k_.
  • Free static variables: s_. Free static variables means statics which are not local and not members of a class or struct, regardless of scope.

Other Formatting:

  • Single-line standalone comments can use either C++ style or C style. End-of-line comments should always use C++ style. The preferred style for multi-line comments is currently under discussion.
  • Headers and implementation files should be named for the class they declare or implement. In general any file should declare or implement only one class.

Coding Guidance

  • Write modern, idiomatic C++ using the new features of C++11; use the -std= CXXFLAGS value in configure.ac to see if you can use features introduced in C++14 or C++17 (as of this writing you may not). In particular:
    • Prefer curly braces for initializers and auto for variable declarations.
    • Declare and initialize variables in one statement at the point that they're first used.
    • Use templates instead of copying code and especially instead of preprocessor macros.
    • Prefer passing references to pointers.
    • Do not ever pass void* (aka gpointer) in C++ code. Use templates or class hierarchies to enforce type safety.

Framework

Gnucash is a Gtk+ project. It's design is object-oriented. The current object orientation is implemented mostly with Gnome's GObject C-language framework and makes heavy use of GLib. While this is a rich eco-system, it brings with it a huge number of dependencies which makes GnuCash difficult to port to other operating systems. Consequently the developers have decided to migrate all of GnuCash except the GUI to C++. No new GObject or GLib-dependent code should be written; instead use C++, the C++ standard library, and Boost libraries.


Guile and Scheme

Gnucash is partly implemented in a Scheme dialect called Guile. (It was originally written mostly in Guile, but that implementation was largely replaced with C several years ago.) In particular, the reports system and part of the business system are written in Guile. To support that, most of the core "engine" API is wrapped and accessible from Guile.

Graphical User Interface

See GUI Guidelines‎‎.