Training Systems for the 21st Century (TS21)

C++ Coding Standard

Last Modified: January 30, 2012

 
 
 

Table of Contents


1.0    Introduction

This document provides the coding standards for the TS21 software integration and development team. The coding standards in this document cover best practices for making code more readable, testable, and maintainable. Standards need to be reviewed and changed as new circumstances arise in software applications and design philosophy. Standards may be waived with the approval of the lead software engineer and the appropriate accompanying documentation. All violations shall be documented and discussed in code inspections. It is important to realize that these standards apply as long as the Trick environment allows it. Any given version of Trick may require deviations of this standard due to compatibility issues between the C++ language and the constructs that Trick allows the project to use.

1.1    The Importance of Standardization

The primary goal of applying these standards is to enable others to use and maintain your code.  It is necessary to put yourself in the perspective of the user looking at your code.  Is this person going to find the information that is expected?  Will it be clear what algorithms you used, and why you used them?  Is your code easily reusable?

1.1.1    Good Points

When a project tries to adhere to common standards a few good things happen: Programs coded according to a standard are more likely to be:

1.1.2    Discussion

The experience of many projects leads to the conclusion that using coding standards makes the project go smoother. Are standards necessary for success? Of course not. But they help, and we need all the help we can get!


1.2    Interpretation

1.2.1    Conventions

The use of the word "shall" in this document requires that any project using this document must comply with the stated standards.  There are no exceptions.  Code that does not conform to a "shall"-requirement will not be accepted into a formal release.

The use of the word "should" is intended to allow wiggle room for exceptions to a requirement.  Any noncompliance to a "should"-requirement must be based upon sound justification and can be challenged/overturned in peer review and standards compliance audits.

The use of the word "may" designates optional requirements.

1.2.2    Terminology

For the sake of simplicity, the use of the word "compiler" means compiler or translator.

"C++ Coding Standard" refers to this document whereas "C++ ANSI Standard" refers to the standard C++ language definition.

 The term "method" means function. Usually, a method is a function that is a member of a class.


 

1.3    Standards Enforcement

First, any serious concerns about the standard should be brought up and worked out within the group. Maybe the standard is not quite appropriate for your situation. It may have overlooked important issues or maybe someone in power vehemently disagrees with certain issues.

It should be understood by the reader that the standards documented here are necessarily affected by current or future limitations levied by the Trick environment. Those requirements will supersede these standards when conflict occurs. This document will be updated to reflect those requirements as it becomes necessary.

Adherence to this standard is required. Period.  Code written in C++ will not be accepted into formal customer releases unless that code conforms to this standard.


 

1.4    Waiver Process

There may be cases where adherence to some aspect of the standard is not feasible. In those cases, the rationale should be brought to the team leads for discussion. Isolated exceptions should be described briefly in a code comment. Global exceptions may influence changing the standard.


 

1.5    Reused Code

Code that is developed by other organizations and then reused (integrated) into the TS21 does not need to follow the coding standards listed in this document. Minor updates made to reused code should follow as closely as possible the coding style used by the originators.


2.0    Miscellaneous

2.1    Code Organization

2.1.1    File Names

Justification

2.2    Character Set

Justification


3.0    Names

3.1    General Rules and Conventions

3.1.1    Uniqueness

Justification

3.1.2    Style

3.1.3    Use Meaningful Names

Example

   // Notice how redundant "stack" becomes. 
   //There's no need to use the word "stack" in methods and variables.

   template <Type>

   class Stack {

      public:

              int stackSize();
              void addItemToStack(Type item);

         ...

   }

   Stack myStack;
   myStack.addItemToStack(4);
   int tmp = myStack.stackSize;

3.1.4    Name Length Policy

Justification

3.1.5    Use of Suffixes and Prefixes

Justification

Examples

If one variable is called then another variable should be called rather than

3.1.6    Use of Abbreviations

Justification

Example

   class FluidOz             // NOT FluidOZ
   class NetworkAbcKey       // NOT NetworkABCKey


3.2    Class Names

Justification

Example

   class NameOneTwo
   class Name

3.3    Class Library Names

Example

A set of X-Windows classes could use Xw as a prefix, so a class name could be:
   class XwMenuBar {
      ...
   }

3.4    Method Names

Justification

Example

   class XwMenuBar {

      public:

         void      addMenu( void );
         void      addMenuOption( int menuNum );
      ...
   }

3.5    Class Attribute Names

Justification

Example

   class XwMenuBar {

      public:

         void      addMenu();
         void      addMenuOption( int menuNum );

      private:

         Menu      mMenu[maxMenus];
         int       mNumMenus;
         ...
   }

3.6    Method Argument Names

Justification

Example

   class NameOneTwo {

      public:

         int  startYourEngines( Engine someEngine, 
                                Engine anotherEngine);
         ...
   }

3.7    Method Argument Lists

Justification

Example

   class Matrix {

      public:

         void  dotProduct( Vector &result, Vector left, Vector right);

         ...
   }

3.8    Global Constants

Justification

Example

    const int A_GLOBAL_CONSTANT = 5;

3.9    Type Names

Justification

Example

   typedef uint16  ModuleType;
   typedef uint32  SystemType;


3.10    Enum Names

Justification

Example

   enum PinStateType {
      PIN_OFF = 0,
      PIN_ON = 1
   }

3.11    Macro Names

Justification

Example

  #define TS_MAX( a, b ) blah
  #define TS_IS_ERR( err ) blah

3.12    C Function Names

Justification


3.13    Namespaces

Justification

Example

  namespace Vendor1 {
      class String { ... };
  }

  namespace Vendor2 {
      class String { ... };
  }

  ...

  using namespace Vendor1;
  using namespace Vendor2;

  Vendor1::String s1;
  Vendor2::String s2;

  ...


4.0    Layout

4.1    Code Indentation Policy

Justification

Example

   void func( void ){

       if (something bad) {

          if (another thing bad) {

              while (more input) {
              }
          }
        }
  }


 

4.2    Braces Policies

4.2.1    Keyword Braces Policies

4.2.2    Function Braces Policies

Justification

Example

   Option 1: 

   void myFunc( void )
   {
   ...
   }

   Option 2:

   void myFunc( void ){
   ...
   }

4.2.3    Class, Structure, and Enumeration Braces Policy

Example

   class MyClass{ 

       void myFunc( void )
       {
       ...
       }

   }

   enum EventModeEnum {

     MODE_SINGLE_EVENT = 1,
     MODE_MULTIPLE_EVENTS = 2

   }


4.3    Alignment Policies

Example

   DWord       dWord;
   DWord*      myDWord;
   char*       myChar;
   char        anotherChar;

   dWord        = 0;
   myDWword     = NULL;
   myChar       = NULL;
   anotherChar  = 0;

   if ( ('0' <= myChar && myChar <= '9') || 
        ('a' <= myChar && myChar <= 'z') || 
        ('A' <= myChar && myChar <= 'Z') ) {
   ...
   }

   scale = (someObject[someIndex].maxX() - someObject[someIndex].minX) \ (plotWin.maxX() - plotWin.minX());

   void myFunction( double arg1, double arg2, ...double argn ){
   ...
   }

4.4    Comment Policies

4.4.1    Endline Comment Policy

Justification

Example

   double interpolate( double* xArray,   // Array of x-values
                       double* yArray,   // Array of y-values
                       double  xVal      // Value to interpolate to
                      )
  {
   ...
  }

   class ViewPort {

   public:
      ...

   private:
      int mXorigin;                   // Screen x-coordinate of viewport
      int mYorigin;                   // Screen y-coordinate of viewport
      int mWid;                       // Width of viewport in pixels
      int mHgt;                       // Height of viewport in pixels
      int mDt;                        // (--) Ok for Trick, not seen by doxygen
      int mTolerance;                 /**< (--) Ok for Trick AND doxygen
      ...
  }

4.4.2    Comment Line Policy

The following policies apply to comments within the code, not including comment blocks for source and header files or for functions/methods.

Justification

Example

   void setupWindow( void )
  {
       Coord      x,y,hotx,hoty;
       Dimension  width, height;
       BitMap     user_bitmap;

       // prompt user for environmental settings
       getWindowSettings( x,y,width,height );
       getCursorSettings( hotx,hoty,user_bitmap );

       // set window positions and size
       window.x      = x;
       window.y      = y;
       window.width  = width;
       window.height = height;
       ...

4.4.3    Function/Method Comment Policy

4.4.4    Source File Comment Policy

4.4.5    Header File Comment Policy

4.4.6    Header File Comment Policy


4.5    Maximum Characters Per Line

Justification


4.6    If Then Else Structure

Justification

Example

    if (numArrays > 3) {

    } 

    else if (0 == numArrays) {

    ...

    } 

    else {

    ...

    }


4.7    Switch Formatting

Example

   switch (...) {

      case 1:
         ...
         // FALL THROUGH

      case 2:
         {        
            int v;
            ...
         }
         break;

      default:
         ...
         break;
   }

4.8    While Formatting

Justification


4.9    Statements Per Line


4.10    Source File Layout


4.11    Header File Layout Policy


4.12    Class Layout Policy

Justification

Example

class LinkedList {

   public:

      // Lifecycle
      LinkedList( void );
      ~LinkedList( void );

      // Access Methods
      Ptr getFirst( void );
      Ptr getLast( void );
      Ptr getCurrent( void );
      ...

      // Operations
      void sort( (int *userFct)(Ptr, Ptr) );

   private:

      // Link Structure
      struct Link {
         Link* prev;
         Link* curr;
         Link* next;
      }

      // Data Members
      Link* first;
      Link* last;
      Link* current;
}



5.0    Design Guidelines

5.1  Usage Policy for goto,continue, break and ?:

5.1.1    Use of Goto

Justification

5.1.2    Use of Continue and Break

Justification

5.1.3    Use of ?:

Justification

5.1.4    Keywords to Avoid

The following keywords should be avoided. Care must be taken when using these keywords as better options are often available.

Justification


5.2    Error Return Check Policy



5.3    Object Constructor Usage Policy

Justification

Example

   class Noise
   {
     public:

       Noise();
       Noise(std::map curveFitConstants);

       double calculateScaleFactor();
       double getScaleFactor() const;

     private:

       double scaleFactor;
       void init(std::map curveFitConstants) throw ();
       .
       .
       .

   };

   Noise::Noise(map curveFitConstants)
   {
     try
     {
       init(curveFitConstants);
     }
     catch(InitializationException)
     {
       // If the initialization was not done correctly, just call the default constructor that
       // sets up reasonable values.
       Noise();

       //Log that an exception has occurred and corrective action taken.
       send_hs_msg("Initialization exception occurred. Setting to default constructor values.", "Noise Class");
     }
     catch()
     {
       // Catch the exception that occurred for some unknown reason.
       Noise();

       //Log that an exception has occurred and corrective action taken.
       send_hs_msg("Unknown exception occurred in Constructor. Setting to default constructor values.", "Noise Class");
     }

     }

   void Noise::init(std::map curveFitConstants) throw ()
   {

     // Initialize the scale factor to 0, so no noise is the default.
     this->scaleFactor = 0.0;

     this->s = curveFitConstants["s"];

     // Check to see if the element was initialized. If not, throw an exception.
     if(abs(s) < minTolerance )
     {
       throw exception();
     }
     .
     .
     .


5.4    General Exceptions

5.4.1   General Exception Handling

5.4.2   Destructor Exceptions


5.5    Make Functions Reentrant

Justification


5.6    Usage Policy for Enumerations, Constants, #defines

Justification

Example

   enum EventMode {
     MODE_SINGLE_EVENT = 1,
     MODE_MULTIPLE_EVENTS = 2
   };
   ...
   EventMode currEventMode;
   ...
   if ( MODE_SINGLE_EVENT == currEventMode ) {
      ...
   }
  class Variables {

   public:

      static const int  A_VARIABLE;
      static const int  B_VARIABLE;
      static const int  C_VARIABLE;
      ...
}

5.7    Macro Function Policy

Justification

Example

   #define  MAX(x,y)       (((x) > (y) ? (x) : (y))        // Get the maximum
   The macro above can be replaced for integers with the following inline    function with no loss of efficiency:
   inline int max( int x, int y ) {
     return (x > y ? x : y);
}

5.8    Variable Initialization Policy

Justification


5.9    Idiom for Initializing Objects


5.10    Braces for Conditional Code

Justification

Example

   // DON'T DO IT THIS WAY - no braces enclosing "exit" 
   if (-1 == ret_val )
      exit( -1 );

   // DO IT THIS WAY
   if ( -1 == ret_val ) {
      exit( -1 );
   }

   // DO IT THIS WAY
   factorial = 1;
   for (i=1; i<n; i++) {
      factorial = factorial * i;
   }

5.11    Function/Method Length Policy

Justification


5.12    Document Null Statements

Example

   // Find the first nonblank character in the label
   i = 0;
   while (' ' == label[i++] ) {
      // NULL
   }

5.13    Embedded Assignments

      while ((c = getchar()) != EOF) {

         // Process the character
         ...
      }

      if ( NULL == (f = fopen("myfile","w")) ) {

         // Print error message and exit
         ...
      }
      printf( "%d %d\n", ++i, i+2 );

5.14    The Law of Demeter

Justification

Caveat

Example

   class SunWorkstation {

     public:

        ...
        SoundCard     sound;
        ...

        void          upVolume( int amount ) { sound.Up( amount ); }
        ...

     private:

        ...

        GraphicsCard  graphics;
   }

   SunWorkstation sun;

   sun.upVolume(1);    // DO
   sun.sound.Up(1);    // DON'T !!!

By making SoundCard private, the "DON'T" will not be possible and will satisfy the Law of Demeter.

5.15    Liskov's Substitution Principle (LSP)

Justification

Example

  We define a rectangle class:

   class Rectangle
   {

     public:

         void setWidth(double w) {itsWidth=w;}
         void setHeight(double h) {itsHeight=w;}
         double getHeight() const {return itsHeight;}
         double getWidth() const {return itsWidth;}

     private:

        double itsWidth;
        double itsHeight;

  };

   Later, we find that we need a square. A square is a special kind of rectangle where 
   the height and width are equal. We can inherit from the class Rectangle, but override the height 
   and width method so that when we change one, we change the other. Because we inherit from rectangle, 
   we get the other methods from class Square: 

   class Square : public Rectangle

   {

     public:

         void setWidth(double w);
         void setHeight(double h);

   };

   void Square::setWidth(double w)
  {

     Rectangle::setWidth(w);
     Rectangle::setHeight(w);

   }

   void Square::setHeight(double h)
   {

     Rectangle::setHeight(h);
     Rectangle::setWidth(h);

   }

   Now, when someone sets the width of a Square object, its height will 
   change correspondingly. And when someone sets its height, the width will change 
   with it. Thus, the invariants of the Square remain intact. The Square object will 
   remain a mathematically proper square.

   Square s;

   s.setWidth(1); // Fortunately sets the height to 1 too.
   s.setHeight(2); // sets width and heigth to 2, good thing.

  But consider the following function:

   void f(Rectangle& r)
   {

     r.setWidth(32); // calls Rectangle::setWidth

   }

   If we pass a reference to a Square object into this function, the Square object
   will be corrupted because the height won't be changed. This is a violation of LSP
   because we have changed the behavior of the base class in the derived class.



5.16    Open/Closed Principle

Justification

Example

  An example of Open/Closed violation:

   // Open-Close Principle - Bad example

   class GraphicEditor
   {

     public:

         void drawShape(Shape s) {...}
         void drawCircle(Circle r) {...}
         void drawRectangle(Rectangle r) {...}

   };

   class Shape
   {

     protected:

         int mType;

     public:

        int getMType();

   }; 

   class Rectangle : Shape
   {

     public:

         Rectangle() { Shape::mType = 1; }

         void drawRectangle();

   }; 

   class Circle : Shape
  {

     public:

       Circle() { Shape::mType = 2; }

   };
   The GraphicEditor class has to be modified for every new shape class that has to be added. There are several disadvantages:

   An example that doesn't violate Open/Close principle:

   // Open-Close Principle - Good example
   class GraphicEditor
   {

     public:

       virtual ~GraphicEditor();
       void drawShape(Shape* s) { s->draw(); }

   };

   class Shape
   {

     protected:

       int mType;

     public:

       int getMType();

       virtual void draw();

   };

   class Rectangle : Shape
   {

     public:

       Rectangle() { Shape::mType = 1; }
       void draw() { ; /*draw the rectangle*/ 

   };

   Rectangle* myRectangle;
   GraphicEditor ge;

   ge.drawShape(myRectangle);
   In the new design draw() is an abstract method used by GraphicEditor for drawing objects, 
   while moving the implementation in the concrete shape objects. Using the Open/Close Principle 
   the problems from the previous design are avoided, because GraphicEditor is not changed when 
   a new shape class is added:

5.17    Operator Overloading Policy

Justification


5.18    Class Implementation Policy

      ClassName.hh
      ClassName.cpp   // or whatever the extension is: cpp, c++
      ClassName_section.cpp

5.19    Const Policy

Example

   void foo(std::string& s);
 
   void bar(std::string const& s)
   {
     foo(s);                      // Compile-time Error since s is const
 
     std::string localCopy = s;
     foo(localCopy);              // OK since localCopy is not const
   } 

   Changes that foo() makes are made to the localCopy object that is local to bar(). 
   In particular, no changes will be made to the const parameter that was passed by reference to bar(). 

5.20    Streams Policy

Justification


5.21    Function/Method Argument Policy

Justification


5.22    Accessor Style Policy

Access methods provide access to the physical or logical attributes of an object.  Accessing an object's attributes directly as we do for C structures is greatly discouraged in C++.  We disallow direct access to attributes to break dependencies, the reason we do most things.  Directly accessing an attribute exposes implementation details about the object.

To see why, ask yourself the following questions:

  1. What if the object decided to provide the attribute in a way other than physical containment?
  2. What if it had to do a database lookup for the attribute?
  3. What if a different object now contained the attribute?
An object makes a contract with the user to provide access to a particular attribute; it should not promise those attributes get their values.  Accessing a physical attribute makes such a promise. By allowing the value of an attribute to be changed in an unexpected way, code may break.
   class X {

      public:

         // Get the age
         int getAge( void )
         {
            return age;
         }

         // Set the age
         void setAge( int age )
         {
            this->age = age;
         }

      private:
         int age;
   }
   class Y {

      public:

         // Get the age
         int getAge( void )
         {
            return age;
         }

         // Set the age
         void setAge( int age )
         {
            this->age = age;
         }

         bool isAMinor( )
         {
            bool result = false;
            if (this->age < 18)
            {
                 result = true;
            }
           
            return result;
         }

      private:

         int age;
   }

5.23    Magic Numbers Policy

Justification

Example

Using magic numbers:
   for (i=0; i<16; i++) {
      Color[i] = allocateColor( i );
   } 
   ... 
   setLineColor( 3 );
   drawLine( x1, y1, x2, y2 );
Replacing magic numbers with constants and enumerations:
   const int MAX_COLORS = 16;

   enum ColorType {
      COLOR_BLACK = 0,
      COLOR_BLUE = 1,
      COLOR_GREEN = 2,
      COLOR_RED = 3,
      ...
      COLOR_WHITE = 10
   };

   for (i=0; i<MAX_COLORS; i++) {
      Color[i] = allocateColor( i );
   }
   ...
   setLineColor( COLOR_RED );
   drawLine( x1, y1, x2, y2 );

5.24    Required Class Elements

Justification


5.25    Virtual Methods

Justification

Example

   class Eps {

      public:

         virtual int getId( );
   };

   class Battery: public Eps {

      public:

         virtual int getId( );
   };

5.26    Data Files


5.27    Interfaces

Justification


5.28    Templates

Justification

Example

   template <class SomeType>

      ClassName& ClassName::operater=(const ClassName& objVar)
      {
         if ( &objVar != this )
         {
         ...


5.29    Null Pointer Checks

Justification

Example

Null Pointer Check:
 

   if ( !fan ) {
     CMN_hs_ptr_msg("ECLSS", "fan");
     // Do any necessary cleanup
     return;
   }
Pass By Reference Vs. Pass by Pointer:
  void foo(SomeObject& obj)
  {
    obj.doSomething();
  }


   void foo(SomeObject* obj)
   {
     obj->doSomething();
   }

5.30    Division Safety

Justification

Example

   if ( myStuff->voltage >= myStuff->minVoltage ) {
     myStuff ->current = myStuff ->power / myStuff ->voltage;
   }
   else {
   ...

5.31    Min/Max Value Checking

Justification

Example

   To limit some variable (rh) in a range where MAX_RH & MIN_RH are constants: 

   // limit relative humidity to range between MIN & MAX
   rh = fmin( fmax(rh, MIN_RH), MAX_RH ); 

   To limit a variable (totalMoles) from going to 0: 
   // limit total moles to small number to prevent divide by zero
   totalMoles = fmax( totalMoles, ECLSS_MIN_GAS_PROPERTY ); 
+
+
+   if ( 1 > initData.numSegments || initData.numSegments > HxFlowPath::MaxSegments) 
+   {
+     ... 
+   }


5.32    Private Data in Trick

Justification

Example

   #include "ts_models/plumbing/fluid/PolyFluid.hh"

   ClassName
   {
     TS_MAKE_SIM_COMPATIBLE( ClassName );
    
      ...
   }


5.33    Relative Include Paths

Justification

Example

   // In same directory
   #include "Foo.hh"

   // In other ts_models directory
   #include "sim_utilities/hs/TsHsMsg.hh"

   // Top level of repository
   #include "cxtf/models/cmn/misc/include/cxtf_class.hh"


5.34    Namespaces

Justification

Example

   class Noise
   {
      public:

       Noise(std::map<string, double> curveFitConstants);

         . . .

     private:

       double scaleFactor;
         . . .

       void init(std::map< string, double> curveFitConstants) throw ();

   };



5.35    Inline Functions

Justification

Example

   class Noise
   {
      public:

       Noise(std::map<string, double> curveFitConstants);

         . . .

       double getScaleFactor() const {
         return this->scaleFactor;
       }

     private:

       double scaleFactor;
         . . .

   };



  • [1]
  • [2]
  • [3]
  • [4]
  • [5]
  • [6]
  • [7]
  • [8]
  • [9]
  • [10]
  • [11]
  • [12]
  • [13]
  • [14]
  • [15]
  • [16]
  • [17]
  • [18]
  • [19]
  • [20]
  • [21]
  • [22]
  • [23]
  • [24]
  • [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]
  • [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]


  • Change Log