JerryScript Coding Standards

This text is a brief overview of JerryScript Coding Standards. Each rule starts with a short description followed by several examples. We believe examples are better than long explanations. Please follow these guidelines when you submit a patch for review.

General rules

  • Indentation is two spaces.
  • Tab characters are not allowed.
  • Maximum line length is 120 characters (excluding newline).
  • No trailing white space is allowed.
  • Run tools/run-tests.py --check-format to check several of the coding conventions automatically.

Comments

Only block comments (/* */) are allowed in JerryScript. Comments should be complete sentences (e.g. start with an upper case letter), except for return value, field and argument descriptions (see the exceptions below). The text part of a comment should focus on explaining why the code is doing something rather than what the code is doing.

+++ Good +++
  /* A perfect comment. */

  /* A perfect multiline
   * comment. Each line should
   * start with an asterisk. */
--- Bad ---
  // Double slash comments are not allowed.

  /* not a complete sentence */

  /* A bad multiline
     comment. */

All types, constants and functions require a description in JerryScript. These comments should start with /**. The starting /** and ending */ must be on separate lines.

+++ Good +++
/**
 * A correct description.
 */
--- Bad ---
/** An incorrect description. */

Preprocessor defines

The name of a preprocessor macro must be an uppercase string and these macros must be preceded by a description. Abbreviations are allowed but not preferred in new code.

+++ Good +++
/**
 * Short description about the constant.
 */
#define JERRY_VALUE_SEVEN 7

/**
 * Short description about the macro function.
 */
#define JERRY_ADD_TWO_NUMBERS(arg1, arg2) \
  ((arg1) + (arg2))

/**
 * Although this is correct, a reviewer might request
 * to change NUMS to NUMBERS. Hence it is recommended
 * to use NUMBERS in the first place.
 */
#define JERRY_ADD_TWO_NUMS(arg1, arg2) \
  ((arg1) + (arg2))
--- Bad ---
#define JERRY_CONSTANT_WITHOUT_DESCRIPTION 5

#define JeRrY_mIxEd_CaSe_NaMe "str"

Conditional preprocessor directives

A comment is required after #else and #endif in JerryScript. The defined keyword should be omitted from these comments.

+++ Good +++
#ifdef JERRY_A

#else /* !JERRY_A */

#endif /* JERRY_A */

#ifdef JERRY_A
#if defined JERRY_B && defined JERRY_C && (JERRY_C > 6)

#else /* !(JERRY_B && JERRY_C && (JERRY_C > 6)) */

#endif /* JERRY_B && JERRY_C && (JERRY_C > 6) */
#endif /* JERRY_A */
--- Bad ---
#ifdef JERRY_A

#endif

#ifdef JERRY_A

#endif /* defined JERRY_A */

#ifdef JERRY_B
  /* Missing comment after else. */
#else

#endif /* JERRY_B */

Code blocks

Each code block must be enclosed in curly braces even if it is a single line statement. These braces must be on separate lines. There must be a single space before the opening parenthesis of the expression after if/while/switch keywords.

+++ Good +++
if (value > 6)
{
  function_call ();
}

if (value > 1)
{
  function_call_a ();
}
else
{
  function_call_b ();
}

do
{
  function_call ();
  value++;
}
while (value < 6);

switch (value)
{
  case A:
  {
    /* FALLTHRU comment is allowed if a
     * switch-case is not terminated by
     * break/continue/return. */

    /* FALLTHRU */
  }
  case B:
  case C:
  {
    break;
  }
  case D:
  {
    /* We can use continue if we are in a loop. */
    continue;
  }
  default:
  {
    return;
  }
}
--- Bad ---
if (value > 6)
  function_call_a ();
else
  function_call_b ();

if (value > 6) {
  function_call_a ();
}

if (value > 6) function_call_a ();
else { function_call_b (); }

if
(value > 6)
{
}

switch (value) {
  case 0: break;
  default: {
    return 5;
  }
}

switch (value)
{
  case A:
  {
    if (value > 6)
    {
      CASE B:
      {
        /* This is allowed in C but
         * not in JerryScript. */
        break;
      }
    }
  }
}

do
  value++;
while (value < 5);

do {
  value++;
} while (value < 5);

do
{
  value++;
} while (value < 5);

Newlines

A newline in JerryScript is a separator which separates different parts of the source code. Its primary purpose is to improve readability. Unlike other rules developers have some freedom to add newlines to their code. However there are some rules.

  • Only a single newline separator is allowed.
  • Source files must be terminated by a newline.
  • Global declarations must be separated by a newline.
  • Newlines are not allowed after an opening curly brace or before a closing curly brace
  • No newlines are allowed between control statements (if-else, while, for, switch, etc.) and their code blocks.
  • There should be a newline after the variable declarations if they are the first statements of a block.
+++ Good +++
if (a > 5)
{
  /* Newline must be present after the first
   * variable declarations of a code block. */
  int j = a - 1;
  int k = a * 2;

  return j + k;
}

while (a < 5)
{
  a++;

  /* It is recommended to put a newline after
   * intermediate variable declarations. */
  int i = a * 2;

  b = i - 3;
}

/* It is a recommended to put newlines around asserts. */
a = b + 5;

JERRY_ASSERT (a < 20);

c = a + 7;

/* It is a good practice to put a newline after a multiline
 * function call (see Function calls later). */
f (a,
   b,
   c);

a = 6;
--- Bad ---
/* No newlines are allowed after an opening curly
 * brace or before a closing curly brace */

while (a > 0)
{

  a = 6;

}

if (a > 5)
{
  while (b < 6)
  {

    b++;
  }

}


/* Two or more newlines are not allowed. */
a = 6;


b = 7;

/* No newlines are allowed between control statements
 * and their code blocks. */

if (a > 6)

{
}

else

{
}

do

{
}

while (a < 6);

Expressions

Spaces are required around binary operators. No space is needed otherwise.

+++ Good +++
a = b + c;
a = (b + c) << 3;
a = b = c + ~d;
a += ++c + d++;
call_function (a * (b + !!c) - d + (e % f));
if (a)
{
}
--- Bad ---
a=b+c;
a = b+c;
a  +=  c  +  ( d );
/* Introduce temporary variables or macros
 * if the expression is too long. Occurs rarely.. */
a = b
  + c;
if ( a + b > 0 )
{
}

Logical operators

All expressions with && and || logical operators must be enclosed in parentheses. A single and multiline form is allowed for these expressions. In the latter case each line must start with the logical operator and each line must be aligned to the column right after the opening parenthesis.

+++ Good +++
/* Single line form. */
a = ((c && d) || (e && f));

a = (c
     && d);

a = (c
     && (d || e)
     && f);

do
{
}
while (a
       && b);

/* This form is rarely used but it is ok. */
if (a
    && (b
        || c
        || d)
    && e)
{
}
--- Bad ---
if (a || b ||
    c)
{
}

/* Parentheses are missing. */
a = b || c;

/* Misaligned &&. */
if (a
  && b)
{
}

Ternary conditional operators

A special form of ternary conditional operators are allowed in JerryScript where the ? and : operators are on separate lines in the same column.

+++ Good +++
a = (b ? c
       : d);

/* Single line form is accepted as well. */
a = (b ? c : d);

/* This form is rarely used but it is ok. */
if (a ? b
      : (c ? d
           : e))
{
}
--- Bad ---
a = b ?
    c : d;

while (a ? b
       : c)
{
}

if (a
    ? b
    : c)
{
}

Function calls

There must be a space after the function name. Each argument must be in the same or separated lines. In the former case there must be a space before the next argument and in the latter case all arguments must be aligned to the same column.

+++ Good +++
function_a ();
function_b (a);
function_c (a, b, c);

function_c (a,
            b,
            c);

function_c (a,
            b,
            function_c (a,
                        b,
                        c);
--- Bad ---
/* No space before the opening parenthesis. */
function_f();

function_f (
);

function_g(a);

function_g
  (a
  );

/* Two arguments on the same line. */
function_h (a, b,
            c);

function_h (a,
            b, c);

/* Misaligned arguments. */
function_h (a,
     b,
     c);

Variable declarations

JerryScript is a pure C99 codebase so variable declarations can be anywhere in the code including inside for-loops. It is recommended to declare a variable before the first use.

+++ Good +++
for (int i = 0; i < 10; i++)
{
  int j = i + 1;
  while (j < 10)
  {
    ++j;
  }
}

/* Multiline form of for loops. */
for (int i = 0;
     i < 10;
     i++)
{
}

Type casting

There must be a space after the closing parenthesis of the type cast. Type casting has no multiline form in JerryScript.

+++ Good +++
int a = (int) double_variable;

int a = (int) (long) (float) double_variable;
--- Bad ---
/* Wrong spaces. */
int a = ( int )double_variable;

/* No multiline form. */
int a = (int)
        double_variable;

Pointers and asterisk character

Each pointer in JerryScript must be a lowercase string which is ending with a _p suffix. Furthermore there must be a space before the asterisk character.

+++ Good +++
  int *int_p;

  /* No need to add multiple _p-s for multiple indirections.
   * It is recommended to avoid these constructs using typedef
   * declarations. A reviewer might request such a change. */
  int ***int_p;

  /* This rule applies for type casting as well. */
  char = *(char *) type_p;
--- Bad ---
  /* No _p after the name. */
  int *ptr;

  /* Wrong asterisk position. */
  int* ptr_p;

  char_p = * (char*)type_p;

Types

Each type in JerryScript must be a lowercase string which ends with a _t suffix. Furthermore each type declaration must be preceded by a short description of the type and each field must have a short description as well.

+++ Good +++
/**
 * Short description of the following structure.
 */
typedef struct
{
  /* Field descriptions do not start with capital letters
   * and there is no full stop at the end. */
  field1_t field1; /**< description of field 1 */
  field2_t field2; /**< description of field 2 */

  field_n_t field_n; /**< description of field n */
} structure_name_t;

/**
 * Another integer type.
 */
typedef int jerry_int;
--- Bad ---
typedef struct
{
  field_t field_without_description;
} structure_without_description_t;

typedef struct { int a; } single_line_struct;

typedef
union {
}
wrong_newlines_t;

/*
  * Bad comment format.
   */
typedef
char wrong_newlines_again_t;

Type usage conventions

  • Passing the number of arguments for a function call is always uint32_t
  • String size/length/position related operation should use lit_utf8_size_t
  • Extended objects internal fields must be uint32_t

Function declarations

Function declarations in JerryScript are verbose but this format reduces the maintenance cost and allows faster understanding of the code.

+++ Good +++
/**
 * Short overview about the purpose of this function.
 *
 * A more detailed explanation if needed.
 *
 * Note:
 *   Extra notes if needed.
 *
 * @return short description about the value
 *         returned by the function
 */
return_value_type_t
function_name (argument1, /**< description of argument1 */
               argument2, /**< description of argument2 */
               ...
               argument_n, /**< description of argument n */
{

  /* Function body. */

} /* function_name */
--- Bad ---
static int
print (char *text) /**< description of text argument */
{
  /* Missing comment before the function. */
} /* print */

/**
 * Prints the text received by the function.
 *
 * @return number of characters printed by the function
 */
int print(char *text)
{
  /* No description of text argument. */
  /* Missing comment at the end of the function. */
}