Skip to content

Latest commit

 

History

History
executable file
·
236 lines (166 loc) · 5.15 KB

conventions.md

File metadata and controls

executable file
·
236 lines (166 loc) · 5.15 KB

Code Conventions

English is the official language to use in the code and comments. Also please use the US spelling and not the UK one (ie. colorize, not colourise).

Use only ASCII (Latin-1, ISO-8859-1) characters anywhere in the code, comments or file names, this means no accents and special characters.

For Python, generally follow the PEP-8 guidelines.

Use the supplied .clang-format using clang-format 9.0.1 to format the codebase.

File names

We use snake_case for file and folder names to avoid issues between windows and unix style file systems. Header files use the "h" extension, source files use "cpp".

contrib // OK
renderDelegate.h // WRONG
render_delegate.cpp // OK
render_delegate.cxx // WRONG

Naming conventions

We follow USD's naming conventions.

Classes are named CamelCase, starting with a module specific prefix unless they are not in a header. Struct is used instead of a class when storing related data in a compact way, without visibility or member functions and they only need the module specific prefix if exposed in a header.

// Base library
class HdArnoldLight; // OK
class MyClass; // WRONG
class myClass; // WRONG

struct MyStruct : public HdRenderDelegate {
    // ... 
}; // WRONG

struct MyOtherStruct {
    float a;
    int b;
    double c;
} // OK

Functions are also CamelCase, private and protected functions on classes are prefixed with an _ .

// Header file
void MyFunction(); // OK
void myFunction(); // WRONG
void my_function(); // WRONG

class HdArnoldRenderDelegate {
public:
    void MyFunction(); // OK
private:
    void MyOtherFunction(); // WRONG
    void _MyFunction(); // OK
};

Source only functions and structs are placed in an anonymous namespace to limit symbols to compilation units and source only functions are prefixed with an _ . Anoymous namespaces do NOT increase identation tot he right.

// Source file

namespace {

void MyFunction()
{
    // ...
} // WRONG

void _MyFunction()
{
    // ...
} // OK

}

Variables are camelCase, class variables are prefixed with an _ . Struct variables are not prefixed with an _ .

int myVariable; // OK
int my_variable; // WRONG

class HdArnoldRenderDelegate {
private:
    int myVariable; // WRONG
    int _myVariable; // OK
};

struct MyStruct {
    int _myVariable; // WRONG
    int myVariable; // OK
};

Macros and enums are all upper case.

#define UPDATE_UNDEFINED   0
#define UPDATE_CAMERA      1

enum PartitionType
{
    POINTS = 0,
    PRIMITIVES,
    DETAIL
};

No explicit use of the pxr namespace. Use the PXR_NAMESPACE_OPEN_SCOPE, PXR_NAMESPACE_CLOSE_SCOPE and PXR_NAMESPACE_USING_DIRECTIVE macros.

pxr::TfToken myToken; // Not OK

PXR_NAMESPACE_OPEN_SCOPE

TfToken mySecondToken; // OK

PXR_NAMESPACE_CLOSE_SCOPE


PXR_NAMESPACE_USING_DIRECTIVE

TfToken myThirdToken; // OK

Coding style

We use C++11 to follow USD's choice of standard and respect the C++ core guidelines.

Some highlights.

Maximum column width for source files is 120. Use short lines (~80 to follow USD's convention), unless indenting becomes hard to follow.

There is no space between the name of a method or function and its parenthesis, neither between the parenthesis and the arguments.

myFunction(a, b);    // OK
myFunction (a, b);   // wrong
myFunction( a, b );  // wrong

End source files with an empty line.

Use const and constexpr wherever possible.

float a = 5.0f; // WRONG if a does not change.
const std::vector<float> vec { 5.0f, 3.0f }; // OK
const float b = 5.0f; // OK
constexpr float b = 5.0f; // OK

Use nullptr in place of NULL or 0 for pointers.

A* a = nullptr; // OK
B* b = 0; // WRONG
C* c = NULL; // WRONG

Use override for class functions.

class Parent {
public:
    virtual void MyFunc() = 0;
    virtual void MyOtherFunc() = 0;
};

class Child : public Parent {
public:
    void MyFunc() override; // OK
    void MyOtherFunc(); // WRONG
};

Use C++ style casting. Use the appropiate casting each occasion, ie. static/reinterpret. const_cast is forbidden unless it is due to a design issue in an external library.

constexpr int a = 5;
constexpr float b = (int) a; // WRONG
constexpr float c = static_cast<int>(a); // OK

Use C++ notation for include files.

#include <cstring>   // OK
#include <string.h>  // WRONG

Use C++ comments (//), not C comments (/* ... \*\/). There must be a space following the //.

// I am a nice comment.
//But I am not.
/* I'm ugly too. */

Indent with four (4) real spaces, no tabs.

Brackets are always required for every block of code.

if (statement)
    OneLineFunction(); // WRONG

if (statement) {
    OneLineFunction(); // OK
}

#define MY_SILLY_MACRO() \
DoSomething(); \
DoSomething2();

if (statement)
    MY_SILLY_MACRO(); // WRONG, DoSomething2 will always run

if (statement) {
    MY_SILLY_MACRO(); // OK, both functions only run when statement is true.
}