//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose: Implementation of SHA-1
//
//=============================================================================

#ifndef CHECKSUM_SHA1_H
#define CHECKSUM_SHA1_H

#if defined( _WIN32 )
#pragma once
#endif

/*
	100% free public domain implementation of the SHA-1
	algorithm by Dominik Reichl <dominik.reichl@t-online.de>


	=== Test Vectors (from FIPS PUB 180-1) ===

	SHA1("abc") =
		A9993E36 4706816A BA3E2571 7850C26C 9CD0D89D

	SHA1("abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq") =
		84983E44 1C3BD26E BAAE4AA1 F95129E5 E54670F1

	SHA1(A million repetitions of "a") =
		34AA973C D4C4DAA4 F61EEB2B DBAD2731 6534016F
*/

#if !defined(_MINIMUM_BUILD_)
#include <stdio.h>  // Needed for file access
#if defined( _PS3 )
#include <sys/memory.h>
#else
#include <memory.h>
#endif
#include <string.h> // Needed for strcat and strcpy
#endif

// If you're compiling big endian, just comment out the following line
#define SHA1_LITTLE_ENDIAN

typedef union
{
	unsigned char c[64];
	unsigned long l[16];
} SHA1_WORKSPACE_BLOCK;

// SHA1 hash
const unsigned int k_cubHash = 20;
const unsigned int k_cchHash = 41; // k_cubHash * 2, plus 1 for terminator
#pragma pack( push, 1 )
typedef	unsigned char SHADigest_t[ k_cubHash ];
#pragma pack( pop )

#if !defined(_MINIMUM_BUILD_)
class CSHA1
#else 
class Minimum_CSHA1 
#endif
{
public:
	// Two different formats for ReportHash(...)
	enum
	{
		REPORT_HEX = 0,
		REPORT_DIGIT = 1
	};

	// Constructor and Destructor
#if !defined(_MINIMUM_BUILD_) 
	CSHA1();
	virtual ~CSHA1() ;
#else
	Minimum_CSHA1() ; 
	~Minimum_CSHA1() ;	// no virtual destructor's in the minimal builds !
#endif	

	unsigned long m_state[5];
	unsigned long m_count[2];
	unsigned char m_buffer[64];
	unsigned char m_digest[k_cubHash];

	void Reset();

	// Update the hash value
	void Update(unsigned char *data, unsigned int len);
#if !defined(_MINIMUM_BUILD_) 
	bool HashFile(char *szFileName);
#endif

	// Finalize hash and report
	void Final();
#if !defined(_MINIMUM_BUILD_) 
	void ReportHash(char *szReport, unsigned char uReportType = REPORT_HEX);
#endif
	void GetHash(unsigned char *uDest);

private:
	// Private SHA-1 transformation
	void Transform(unsigned long state[5], unsigned char buffer[64]);

	// Member variables
	unsigned char m_workspace[64];
	SHA1_WORKSPACE_BLOCK *m_block; // SHA1 pointer to the byte array above
};

#define GenerateHash( hash, pubData, cubData ) { CSHA1 sha1; sha1.Update( (byte *)pubData, cubData ); sha1.Final(); sha1.GetHash( hash ); } 

#if !defined(_MINIMUM_BUILD_)
// hash comparison function, for use with CUtlMap/CUtlRBTree
bool HashLessFunc( SHADigest_t const &lhs, SHADigest_t const &rhs );

// utility class for manipulating SHA1 hashes in their compact form
struct CSHA
{
public:
	SHADigest_t m_shaDigest;

	CSHA()
	{
		memset( m_shaDigest, 0, k_cubHash );
	}

	explicit CSHA( const SHADigest_t rhs )
	{
		memcpy( m_shaDigest, rhs, k_cubHash );
	}

	SHADigest_t &SHADigest()
	{
		return m_shaDigest;
	}

	bool operator<( const CSHA &rhs ) const
	{
		return memcmp( m_shaDigest, rhs.m_shaDigest, k_cubHash ) < 0;
	}

	bool operator==( const CSHA &rhs ) const
	{
		return memcmp( m_shaDigest, rhs.m_shaDigest, k_cubHash ) == 0;
	}

	bool operator!=( const CSHA &rhs ) const
	{
		return !(*this == rhs);
	}

	bool operator==( const SHADigest_t &rhs ) const
	{
		return memcmp( m_shaDigest, rhs, k_cubHash ) == 0;
	}

	bool operator!=( const SHADigest_t &rhs ) const
	{
		return !(*this == rhs);
	}

	CSHA &operator=( const SHADigest_t rhs )
	{
		memcpy( m_shaDigest, rhs, k_cubHash );
		return *this;
	}

	void AssignTo( SHADigest_t rhs ) const
	{
		memcpy( rhs, m_shaDigest, k_cubHash );
	}
};
#endif // _MINIMUM_BUILD_

#endif // CHECKSUM_SHA1_H