2013-06-14 00:51:33 +00:00
|
|
|
/* radare - LGPL - Copyright 2009-2013 - pancake */
|
2009-03-31 00:50:02 +00:00
|
|
|
|
|
|
|
#include "r_crypto.h"
|
2009-09-14 23:45:58 +00:00
|
|
|
#include "../config.h"
|
|
|
|
|
2013-08-30 00:59:20 +00:00
|
|
|
R_LIB_VERSION (r_crypto);
|
2013-06-14 00:51:33 +00:00
|
|
|
|
2010-05-25 23:42:22 +00:00
|
|
|
static struct r_crypto_plugin_t *crypto_static_plugins[] =
|
2009-09-14 23:45:58 +00:00
|
|
|
{ R_CRYPTO_STATIC_PLUGINS };
|
|
|
|
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API struct r_crypto_t *r_crypto_init(struct r_crypto_t *cry, int hard) {
|
2009-09-14 23:45:58 +00:00
|
|
|
int i;
|
2010-05-28 00:44:51 +00:00
|
|
|
RCryptoPlugin *static_plugin;
|
|
|
|
|
2009-09-14 23:45:58 +00:00
|
|
|
if (cry) {
|
|
|
|
cry->key = NULL;
|
|
|
|
cry->iv = NULL;
|
|
|
|
cry->key_len = 0;
|
|
|
|
cry->user = NULL;
|
2009-09-15 11:24:28 +00:00
|
|
|
if (hard) {
|
|
|
|
// first call initializes the output_* variables
|
|
|
|
r_crypto_get_output(cry);
|
2010-05-26 16:25:35 +00:00
|
|
|
INIT_LIST_HEAD(&cry->plugins);
|
2010-05-28 00:44:51 +00:00
|
|
|
for(i=0;crypto_static_plugins[i];i++) {
|
|
|
|
static_plugin = R_NEW (RCryptoPlugin);
|
|
|
|
memcpy (static_plugin, crypto_static_plugins[i], sizeof (RCryptoPlugin));
|
|
|
|
r_crypto_add(cry, static_plugin);
|
|
|
|
}
|
2009-09-15 11:24:28 +00:00
|
|
|
}
|
2009-09-14 23:45:58 +00:00
|
|
|
}
|
|
|
|
return cry;
|
|
|
|
}
|
|
|
|
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API int r_crypto_add(RCrypto *cry, struct r_crypto_plugin_t *h) {
|
2009-09-15 11:24:28 +00:00
|
|
|
// add a check ?
|
2010-05-26 16:25:35 +00:00
|
|
|
list_add_tail(&(h->list), &(cry->plugins));
|
2009-09-14 23:45:58 +00:00
|
|
|
return R_TRUE;
|
|
|
|
}
|
|
|
|
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API int r_crypto_del(RCrypto *cry, struct r_crypto_plugin_t *h) {
|
2009-09-15 11:24:28 +00:00
|
|
|
list_del(&(h->list));
|
|
|
|
return R_TRUE;
|
|
|
|
}
|
2009-09-14 23:45:58 +00:00
|
|
|
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API struct r_crypto_t *r_crypto_new() {
|
|
|
|
RCrypto *cry = R_NEW(RCrypto);
|
|
|
|
return r_crypto_init (cry, R_TRUE);
|
2009-09-15 11:24:28 +00:00
|
|
|
}
|
|
|
|
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API struct r_crypto_t *r_crypto_as_new(struct r_crypto_t *cry) {
|
|
|
|
RCrypto *c = R_NEW(RCrypto);
|
2009-09-15 11:24:28 +00:00
|
|
|
if (c != NULL) {
|
2013-08-30 00:59:20 +00:00
|
|
|
r_crypto_init (c, R_FALSE); // soft init
|
|
|
|
memcpy (&c->plugins, &cry->plugins, sizeof (cry->plugins));
|
2009-09-15 11:24:28 +00:00
|
|
|
}
|
|
|
|
return c;
|
2009-09-14 23:45:58 +00:00
|
|
|
}
|
|
|
|
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API struct r_crypto_t *r_crypto_free(RCrypto *cry) {
|
2009-09-14 23:45:58 +00:00
|
|
|
// TODO: call the destructor function of the plugin to destory the *user pointer if needed
|
|
|
|
// TODO: free plugins
|
|
|
|
free(cry->output);
|
|
|
|
free(cry->key);
|
|
|
|
free(cry->iv);
|
|
|
|
free(cry);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API int r_crypto_use(RCrypto *cry, const char *algo) {
|
2009-09-14 23:45:58 +00:00
|
|
|
int ret = R_FALSE;
|
|
|
|
struct list_head *pos;
|
2010-05-26 16:25:35 +00:00
|
|
|
list_for_each_prev(pos, &cry->plugins) {
|
2010-05-25 23:42:22 +00:00
|
|
|
struct r_crypto_plugin_t *h = list_entry(pos, struct r_crypto_plugin_t, list);
|
2009-09-14 23:45:58 +00:00
|
|
|
if (h->use(algo)) {
|
|
|
|
cry->h = h;
|
|
|
|
cry->key_len = h->get_key_size(cry);
|
|
|
|
cry->key = malloc(cry->key_len);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API int r_crypto_set_key(RCrypto *cry, const ut8* key, int mode, int direction) {
|
2009-09-14 23:45:58 +00:00
|
|
|
int ret = R_FALSE;
|
|
|
|
if (cry->h && cry->h->set_key)
|
|
|
|
ret = cry->h->set_key(cry, key, mode, direction);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API int r_crypto_get_key_size(RCrypto *cry) {
|
2009-09-14 23:45:58 +00:00
|
|
|
int ret = 0;
|
|
|
|
if (cry->h && cry->h->get_key_size)
|
|
|
|
ret = cry->h->get_key_size(cry);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API int r_crypto_set_iv(RCrypto *cry, const ut8 *iv) {
|
2009-09-14 23:45:58 +00:00
|
|
|
int ret = R_FALSE;
|
|
|
|
if (cry->h && cry->h->set_iv)
|
|
|
|
ret = cry->h->set_iv(cry, iv);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// return the number of bytes written in the output buffer
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API int r_crypto_update(RCrypto *cry, ut8 *buf, int len) {
|
2009-09-14 23:45:58 +00:00
|
|
|
int olen = 0; // length of output bytes
|
|
|
|
if (cry->h && cry->h->update)
|
|
|
|
olen = cry->h->update(cry, buf, len);
|
|
|
|
return olen;
|
|
|
|
}
|
|
|
|
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API int r_crypto_final(RCrypto *cry, ut8 *buf, int len) {
|
2009-09-14 23:45:58 +00:00
|
|
|
// TODO: same as update()
|
|
|
|
int olen = 0; // length of output bytes
|
|
|
|
if (cry->h && cry->h->final)
|
|
|
|
olen = cry->h->final(cry, buf, len);
|
|
|
|
return olen;
|
|
|
|
}
|
|
|
|
|
|
|
|
// append data to the output buffer
|
|
|
|
// TODO: internal api?? used from plugins?
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API int r_crypto_append(RCrypto *cry, const ut8 *buf, int len) {
|
2009-09-14 23:45:58 +00:00
|
|
|
if (cry->output_len+len > cry->output_size) {
|
|
|
|
cry->output_size += 4096 + len;
|
|
|
|
cry->output = realloc(cry->output, cry->output_size);
|
|
|
|
}
|
|
|
|
memcpy(cry->output + cry->output_len, buf, len);
|
|
|
|
cry->output_len += len;
|
|
|
|
return cry->output_len;
|
|
|
|
}
|
2009-03-31 00:50:02 +00:00
|
|
|
|
2009-09-14 23:45:58 +00:00
|
|
|
// NOTE: Passes ownership of buffer, coz other is freed
|
2013-08-30 00:59:20 +00:00
|
|
|
R_API ut8 *r_crypto_get_output(RCrypto *cry) {
|
2009-09-14 23:45:58 +00:00
|
|
|
ut8 *buf = cry->output;
|
|
|
|
// free the buffer
|
|
|
|
cry->output_size = 4096;
|
|
|
|
cry->output = malloc(4096);
|
|
|
|
cry->output_len = 0;
|
|
|
|
return buf;
|
2009-03-31 00:50:02 +00:00
|
|
|
}
|