From 22c1a411090acdd252cd20bc5e9cd2bab2280b5d Mon Sep 17 00:00:00 2001 From: antma Date: Wed, 15 Jan 2014 19:29:26 +0400 Subject: [PATCH 1/2] improve portability by removing rdtsc calls https://github.com/vysheng/tg/issues/8 --- mtproto-client.c | 3 +-- mtproto-common.c | 12 ++++++------ mtproto-common.h | 6 ------ 3 files changed, 7 insertions(+), 14 deletions(-) diff --git a/mtproto-client.c b/mtproto-client.c index 844af9c..e8e692c 100644 --- a/mtproto-client.c +++ b/mtproto-client.c @@ -90,14 +90,13 @@ struct connection_methods auth_methods = { }; long long precise_time; -long long precise_time_rdtsc; + double get_utime (int clock_id) { struct timespec T; my_clock_gettime (clock_id, &T); double res = T.tv_sec + (double) T.tv_nsec * 1e-9; if (clock_id == CLOCK_REALTIME) { precise_time = (long long) (res * (1LL << 32)); - precise_time_rdtsc = rdtsc (); } return res; } diff --git a/mtproto-common.c b/mtproto-common.c index 12efcca..991e6d8 100644 --- a/mtproto-common.c +++ b/mtproto-common.c @@ -67,6 +67,9 @@ int get_random_bytes (unsigned char *buf, int n) { } } close (h); + if (r < 0) { + r = 0; + } } if (r < n) { @@ -105,18 +108,15 @@ void my_clock_gettime (int clock_id UU, struct timespec *T) { #endif } - void prng_seed (const char *password_filename, int password_length) { unsigned char *a = talloc0 (64 + password_length); - long long r = rdtsc (); struct timespec T; my_clock_gettime (CLOCK_REALTIME, &T); memcpy (a, &T.tv_sec, 4); - memcpy (a+4, &T.tv_nsec, 4); - memcpy (a+8, &r, 8); + memcpy (a + 4, &T.tv_nsec, 4); unsigned short p = getpid (); - memcpy (a + 16, &p, 2); - int s = get_random_bytes (a + 18, 32) + 18; + memcpy (a + 8, &p, 2); + int s = get_random_bytes (a + 10, 32) + 10; if (password_filename) { int fd = open (password_filename, O_RDONLY); if (fd < 0) { diff --git a/mtproto-common.h b/mtproto-common.h index d15c922..011aed3 100644 --- a/mtproto-common.h +++ b/mtproto-common.h @@ -375,12 +375,6 @@ static inline void fetch_skip_str (void) { fetch_str (l); } -static __inline__ unsigned long long rdtsc(void) { - unsigned hi, lo; - __asm__ __volatile__ ("rdtsc" : "=a"(lo), "=d"(hi)); - return ( (unsigned long long)lo)|( ((unsigned long long)hi)<<32 ); -} - static inline long have_prefetch_ints (void) { return in_end - in_ptr; } From 192b82ef95eb36832b4892446a2f686b74291c22 Mon Sep 17 00:00:00 2001 From: antma Date: Fri, 17 Jan 2014 16:19:12 +0400 Subject: [PATCH 2/2] use rdtsc for seeding prng only at i386 and x86_64 platforms https://github.com/vysheng/tg/issues/8 --- mtproto-common.c | 59 ++++++++++++++++++++++++++++++++++-------------- 1 file changed, 42 insertions(+), 17 deletions(-) diff --git a/mtproto-common.c b/mtproto-common.c index 991e6d8..7ca8bee 100644 --- a/mtproto-common.c +++ b/mtproto-common.c @@ -65,11 +65,10 @@ int get_random_bytes (unsigned char *buf, int n) { if (verbosity >= 3) { logprintf ( "added %d bytes of real entropy to secure random numbers seed\n", r); } - } - close (h); - if (r < 0) { + } else { r = 0; } + close (h); } if (r < n) { @@ -79,13 +78,12 @@ int get_random_bytes (unsigned char *buf, int n) { } int s = read (h, buf + r, n - r); close (h); - if (s < 0) { - return r; + if (s > 0) { + r += s; } - r += s; } - if (r >= (int)sizeof (long)) { + if (r >= (int) sizeof (long)) { *(long *)buf ^= lrand48 (); srand48 (*(long *)buf); } @@ -108,35 +106,62 @@ void my_clock_gettime (int clock_id UU, struct timespec *T) { #endif } +/* RDTSC */ +#if defined(__i386__) +#define HAVE_RDTSC +static __inline__ unsigned long long rdtsc (void) { + unsigned long long int x; + __asm__ volatile ("rdtsc" : "=A" (x)); + return x; +} +#elif defined(__x86_64__) +#define HAVE_RDTSC +static __inline__ unsigned long long rdtsc (void) { + unsigned hi, lo; + __asm__ __volatile__ ("rdtsc" : "=a"(lo), "=d"(hi)); + return ((unsigned long long) lo) | (((unsigned long long) hi) << 32); +} +#endif + void prng_seed (const char *password_filename, int password_length) { - unsigned char *a = talloc0 (64 + password_length); struct timespec T; my_clock_gettime (CLOCK_REALTIME, &T); - memcpy (a, &T.tv_sec, 4); - memcpy (a + 4, &T.tv_nsec, 4); + RAND_add (&T, sizeof (T), 4.0); +#ifdef HAVE_RDTSC + unsigned long long r = rdtsc (); + RAND_add (&r, 8, 4.0); +#endif unsigned short p = getpid (); - memcpy (a + 8, &p, 2); - int s = get_random_bytes (a + 10, 32) + 10; - if (password_filename) { + RAND_add (&p, sizeof (p), 0.0); + p = getppid (); + RAND_add (&p, sizeof (p), 0.0); + unsigned char rb[32]; + int s = get_random_bytes (rb, 32); + if (s > 0) { + RAND_add (rb, s, s); + } + memset (rb, 0, sizeof (rb)); + if (password_filename && password_length > 0) { int fd = open (password_filename, O_RDONLY); if (fd < 0) { logprintf ( "Warning: fail to open password file - \"%s\", %m.\n", password_filename); } else { - int l = read (fd, a + s, password_length); + unsigned char *a = talloc0 (password_length); + int l = read (fd, a, password_length); if (l < 0) { logprintf ( "Warning: fail to read password file - \"%s\", %m.\n", password_filename); } else { if (verbosity > 0) { logprintf ( "read %d bytes from password file.\n", l); } - s += l; + RAND_add (a, l, l); } close (fd); + tfree_secure (a, password_length); } } - RAND_seed (a, s); BN_ctx = BN_CTX_new (); - tfree_secure (a, 64 + password_length); + ensure_ptr (BN_ctx); } int serialize_bignum (BIGNUM *b, char *buffer, int maxlen) {