"This does not contain everything, but what it contains is true".
- Use Doxygen to document all constants, enums, typedefs, struct fields and function declarations. Use the following style for function comments headers:
/**
* @brief Notice that the Doxygen header starts with /**
*
* All function declarations in .h files in the repository are documented
* using Doxygen. Every function has an @brief, lists its @param as well
* as its @return value. In some cases, @warning is used to point out a
* particularly problematic issue with respect to usage or behavior.
*
* @param [in] parameter1 Long descriptions must be indented for easier
* reading. Notice how this description is long but
* is indented with the parameter description to set
* it apart from the next param.
* @param [in] parameter2 Here's that other param.
*
*
* @param [out] parameter3 But when descriptions are short.
* @param [out] parameter4 It's fine for them to omit the tabs.
*
* @warning Notice that the comments in the Doxygen header are one
* space after the * and do not start on the same line as /**.
*/
-
Use snake case (all lowercase, words separated by an underscore) for all identifiers, because this makes for a consistent experience when reading code.
nr_datastore_t ds_type;
-
Add a
_t
postfix to all typedef identifiers, because this makes it clear an identifier is a typedef.typedef struct _newrelic_segment_t newrelic_segment_t;
-
Use bool for values that can only be true or false, because it improves readability and reduces ambiguity.
bool create_successful = false;
-
Prefix all non-static internal functions with
nr_
, and all public functions withnewrelic_
, because this makes it obvious where a function belongs.nr_status_t newrelic_connect_app(newrelic_app_t* app, unsigned short timeout_ms); bool nr_txn_ignore(nrtxn_t* txn);
-
Prefix all internal enum values with
NR_
, and all public enum values withNEWRELIC_
, because this makes it obvious where an enum value belongs.typedef enum _newrelic_loglevel_t { NEWRELIC_LOG_ERROR, NEWRELIC_LOG_WARNING, NEWRELIC_LOG_INFO, NEWRELIC_LOG_DEBUG, } newrelic_loglevel_t; typedef enum _nr_status_t { NR_SUCCESS = 0, NR_FAILURE = -1, } nr_status_t;
-
Document all public functions in header files, because this makes it possible to use a function and know about its caveats without digging into the implementation.
/** * @brief Ignore the current transaction * * Given a transaction, this function instructs the C SDK to not send data to * New Relic for that transaction. * * @param [in] transaction A transaction. * * @return true on success. */ bool newrelic_ignore_transaction(newrelic_txn_t* transaction);
-
Use Yoda conditions, because this reduces the risk of accidental assignment.
if (NULL == attribute) { return; }
-
Use NULL instead of 0 for all null pointer values, because this makes it clear that it's a pointer and not a numeric value.
newrelic_txn_t* transaction = NULL;
-
Use designated struct initialization wherever possible, because it increases readability.
nr_segment_datastore_params_t params = { .operation = segment->type.datastore.operation, .collection = segment->type.datastore.collection, .instance = &segment->type.datastore.instance };
-
Allocate structs on the stack wherever possible, because this avoids more expensive heap allocations.
nr_segment_datastore_params_t params = { .operation = segment->type.datastore.operation, .collection = segment->type.datastore.collection, .instance = &segment->type.datastore.instance }; nr_segment_datastore_end(segment->segment, ¶ms);
-
Avoid vague TODO and FIXME comments, because those never get done.
72877ec1ce (A Dev 1998-09-20 07:15:30 -0700 846) return -1; /* TODO: dodgy */
-
Use const for function parameters passed by reference where the function does not modify the data, because this prevents accidental changes and makes it easier to reason about code parts.
newrelic_custom_event_t* newrelic_create_custom_event(const char* event_type);
-
Format code with clang-format 3.8 using our style configuration file.
-
Disregard directives from this list when other approaches are more beneficial to the overall goal. Do not blindly follow rules.