Skip to content

Latest commit

 

History

History
132 lines (84 loc) · 3.45 KB

Reviewing.md

File metadata and controls

132 lines (84 loc) · 3.45 KB

Checklist for reviewing code.

Reviewing Haskell code

Timing safe equality.

All cryptographically sensitive data types should have timing safe equality comparison. The class Equality is the timing safe counterpart of the Eq class. All such sensitive types should define the Equality instance first using the eq and the monoid instance of the Result type. The Eq type should then be a simple use of (===).

Good example.

data GoodSensitiveType = ...

instance Equality GoodSensitiveType where
	eq a b = ...

instance Eq GoodSensitiveType where
	(==) = (===)

Here are some other example of a good definition.

-- makes use of the Equality instance for pair.
newtype AnotherGoodType = AGT (Foo,Bar) deriving Equality

instance Eq AnotherGoodType where
	(==) = (===)

-- | Makes use of the Equality and Eq instances of tuple types.
newtype Foo = Foo (Tuple 42 Word32) deriving (Equality, Eq)

A bad example. The deriving clause makes the (==) timing dependent even if Foo and Bar have timing safe equality.

data BadSensitiveType = BadSensitiveType Foo Bar deriving Eq

Dangerous modules

We document here some of the Haskell modules that do dangerous stuff so that they can be audited more carefully. The exact dangers are documented in the module.

  1. Raaz.Cipher.ChaCha20.Recommendation.

  2. Raaz.Cipher.ChaCha20.Implementation.CPortable

Reviewing C code.

For speed, the block primitives are written in C. Most primitives have a default word type (64-bit unsigned int for sha512 for example). They also have a block that is essentially an array of such words (for sha512 it is 16. So it will be common to see declarations of the following kind.

typedef uint64_t   Word;  /* basic unit of sha512 hash  */
#define HASH_SIZE  8      /* Number of words in a Hash  */
#define BLOCK_SIZE 16     /* Number of words in a block */

typedef Word Hash [ HASH_SIZE  ];
typedef Word Block[ BLOCK_SIZE ];

In such a setting, we often have a loop that goes over all the blocks. This would typically look like the following.

void foo(Block *ptr, int nblocks, ...)
{
	/* Other stuff here */

	while(nblocks > 0) /* looping over all blocks */
	{

       doSomethingOn((*ptr)[i]); /* do something on the ith word in the current block */

       -- nblocks; ++ ptr; /* move to the next block ensure these are
                            * on the same line
	                        */

	}
}

We follow the above convention because it reduces the chance of incorrect pointer arithmetic. The bugs are concentrated on the definition of the block and word types. So if one is reviewing such low level code, it is better to get familiarised with this programming pattern.

Stuff that effects both C and Haskell.

Alignment and restrict for block primitives

GCC can perform brutal optimisations even to Portable C implementations if the right argument is qualified with restrict and the alignment is matched to the associated vector operations. But these features are Dangerous

  1. Make sure that the alignment that you provide at the Haskell end when you define the Implementation is the same as (or a multiple of) that at the C side.

  2. Since we expect to call these functions as an ffi it is relatively safe to assume that the buffer is not aliased. So restrict can also be given.

Be careful with ChaCha20 implementation where the alignment is also used for the PRG buffer (See the dangerous modules subsection).