/* +----------------------------------------------------------------------+ | PHP version 4.0 | +----------------------------------------------------------------------+ | Copyright (c) 1997, 1998, 1999, 2000 The PHP Group | +----------------------------------------------------------------------+ | This source file is subject to version 2.02 of the PHP license, | | that is bundled with this package in the file LICENSE, and is | | available at through the world-wide-web at | | http://www.php.net/license/2_02.txt. | | If you did not receive a copy of the PHP license and are unable to | | obtain it through the world-wide-web, please send a note to | | license@php.net so we can mail you a copy immediately. | +----------------------------------------------------------------------+ | Authors: Sascha Schumann | | | | HMAC and KEYGEN functionality added by | | Nikos Mavroyanopoulos | +----------------------------------------------------------------------+ */ #include "php.h" #if HAVE_LIBMHASH #include "fcntl.h" #include "php_mhash.h" #include "mhash.h" #include "php_ini.h" #include "php_globals.h" #include "ext/standard/info.h" function_entry mhash_functions[] = { PHP_FE(mhash_get_block_size, NULL) PHP_FE(mhash_get_hash_name, NULL) PHP_FE(mhash_keygen_s2k, NULL) PHP_FE(mhash_count, NULL) PHP_FE(mhash, NULL) {0} , }; static PHP_MINIT_FUNCTION(mhash); zend_module_entry mhash_module_entry = { "mhash", mhash_functions, PHP_MINIT(mhash), NULL, NULL, NULL, NULL, STANDARD_MODULE_PROPERTIES, }; #ifdef COMPILE_DL_MHASH ZEND_GET_MODULE(mhash) #endif #define MHASH_FAILED_MSG "mhash initialization failed" #define MHASH_KEYGEN_FAILED_MSG "mhash key generation failed" static PHP_MINIT_FUNCTION(mhash) { int i; char *name; char buf[128]; for (i = 0; i <= mhash_count(); i++) { name = mhash_get_hash_name(i); if (name) { snprintf(buf, 127, "MHASH_%s", name); zend_register_long_constant(buf, strlen(buf) + 1, i, CONST_PERSISTENT, module_number ELS_CC); free(name); } } return SUCCESS; } /* {{{ proto int mhash_count() get the number of available hashes */ PHP_FUNCTION(mhash_count) { RETURN_LONG(mhash_count()); } /* }}} */ /* {{{ proto int mhash_get_block_size(int hash) get the block size of hash */ PHP_FUNCTION(mhash_get_block_size) { pval **hash; if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &hash) == FAILURE) { WRONG_PARAM_COUNT; } convert_to_long_ex(hash); RETURN_LONG(mhash_get_block_size((*hash)->value.lval)); } /* }}} */ /* {{{ proto string mhash_get_hash_name(int hash) get the name of hash */ PHP_FUNCTION(mhash_get_hash_name) { pval **hash; char *name; if (ZEND_NUM_ARGS() != 1 || zend_get_parameters_ex(1, &hash) == FAILURE) { WRONG_PARAM_COUNT; } convert_to_long_ex(hash); name = mhash_get_hash_name((*hash)->value.lval); if (name) { RETVAL_STRING(name, 1); free(name); } else { RETVAL_FALSE; } } /* }}} */ /* {{{ proto string mhash(int hash, string data, [string key]) hash data with hash */ PHP_FUNCTION(mhash) { pval **hash, **data, **key; MHASH td; int bsize; unsigned char *hash_data; int num_args; num_args = ZEND_NUM_ARGS(); if (num_args < 2 || num_args > 3) { WRONG_PARAM_COUNT; } if (num_args == 2) { /* 2 arguments, just hash */ if (zend_get_parameters_ex(2, &hash, &data) == FAILURE) { WRONG_PARAM_COUNT; } } else { /* 3 arguments, do HMAC hash (keyed hash) */ if (zend_get_parameters_ex(3, &hash, &data, &key) == FAILURE) { WRONG_PARAM_COUNT; } convert_to_string_ex(key); } convert_to_long_ex(hash); convert_to_string_ex(data); bsize = mhash_get_block_size((*hash)->value.lval); if (num_args == 3) { if (mhash_get_hash_pblock((*hash)->value.lval) == 0) { php_error(E_WARNING, MHASH_FAILED_MSG); RETURN_FALSE; } td = mhash_hmac_init((*hash)->value.lval, (*key)->value.str.val, (*key)->value.str.len, mhash_get_hash_pblock((*hash)->value. lval)); } else { td = mhash_init((*hash)->value.lval); } if (td == MHASH_FAILED) { php_error(E_WARNING, MHASH_FAILED_MSG); RETURN_FALSE; } mhash(td, (*data)->value.str.val, (*data)->value.str.len); if (num_args == 3) { hash_data = (unsigned char *) mhash_hmac_end(td); } else { hash_data = (unsigned char *) mhash_end(td); } if (hash_data) { RETVAL_STRINGL(hash_data, bsize, 1); mhash_free(hash_data); } else { RETURN_FALSE; } } /* }}} */ /* {{{ proto string mhash_keygen_s2k(int hash, string input_password, string salt, int bytes) generate a key using hash functions */ /* SALTED S2K uses a fixed salt */ #define SALT_SIZE 8 PHP_FUNCTION(mhash_keygen_s2k) { pval **hash, **input_password, **bytes, **input_salt; unsigned char *key; int password_len, salt_len; int hashid, size=0, val; KEYGEN keystruct; char salt[SALT_SIZE], *ret; char* password, error[128]; if (ZEND_NUM_ARGS() != 4) { WRONG_PARAM_COUNT; } if (zend_get_parameters_ex(4, &hash, &input_password, &input_salt, &bytes) == FAILURE) { WRONG_PARAM_COUNT; } convert_to_long_ex(hash); convert_to_string_ex(input_password); convert_to_string_ex(input_salt); convert_to_long_ex(bytes); password = (*input_password)->value.str.val; password_len = (*input_password)->value.str.len; salt_len = (*input_salt)->value.str.len; if (salt_len > mhash_get_keygen_salt_size(KEYGEN_S2K_SALTED)) { sprintf( error, "The specified salt [%d] is more bytes than the required by the algorithm [%d]\n", salt_len, mhash_get_keygen_salt_size(KEYGEN_S2K_SALTED)); php_error(E_WARNING, error); } memset( salt, 0, SALT_SIZE); memcpy( salt, (*input_salt)->value.str.val, salt_len); salt_len=SALT_SIZE; /* if (salt_len==0) { * php_error(E_WARNING, "Not using salt is really not recommended); * } */ hashid = (*hash)->value.lval; size = (*bytes)->value.lval; keystruct.hash_algorithm[0]=hashid; keystruct.hash_algorithm[1]=hashid; keystruct.count=0; keystruct.salt = salt; keystruct.salt_size = salt_len; ret = malloc(size); if (ret==NULL) { php_error(E_WARNING, MHASH_KEYGEN_FAILED_MSG); RETURN_FALSE; } val = mhash_keygen_ext( KEYGEN_S2K_SALTED, keystruct, ret, size, password, password_len); if ( val >= 0) { RETVAL_STRINGL(ret, size, 1); free(ret); } else { php_error(E_WARNING, MHASH_KEYGEN_FAILED_MSG); free(ret); RETURN_FALSE; } } /* }}} */ #endif