2014-01-10 15:32:57 +04:00
|
|
|
/*
|
|
|
|
This file is part of telegram-client.
|
|
|
|
|
|
|
|
Telegram-client is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
Telegram-client is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this telegram-client. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
Copyright Vitaly Valtman 2013
|
|
|
|
*/
|
|
|
|
|
2014-01-13 17:05:25 +04:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2014-01-10 18:37:56 +04:00
|
|
|
#define _GNU_SOURCE
|
|
|
|
|
2014-01-10 15:32:57 +04:00
|
|
|
#include <assert.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <openssl/err.h>
|
2014-08-14 22:03:33 +04:00
|
|
|
#include <openssl/rand.h>
|
2014-01-10 20:54:47 +04:00
|
|
|
#include <zlib.h>
|
2014-08-14 22:03:33 +04:00
|
|
|
#include <time.h>
|
|
|
|
#include <sys/time.h>
|
2014-01-10 15:32:57 +04:00
|
|
|
|
2014-08-14 22:03:33 +04:00
|
|
|
//#include "interface.h"
|
2014-01-10 15:32:57 +04:00
|
|
|
#include "tools.h"
|
|
|
|
|
2014-01-11 20:00:31 +04:00
|
|
|
#ifdef DEBUG
|
2014-01-13 16:36:29 +04:00
|
|
|
#define RES_PRE 8
|
|
|
|
#define RES_AFTER 8
|
2014-01-11 20:00:31 +04:00
|
|
|
#define MAX_BLOCKS 1000000
|
|
|
|
void *blocks[MAX_BLOCKS];
|
|
|
|
void *free_blocks[MAX_BLOCKS];
|
|
|
|
int used_blocks;
|
|
|
|
int free_blocks_cnt;
|
|
|
|
#endif
|
|
|
|
|
2014-08-14 22:03:33 +04:00
|
|
|
|
|
|
|
void logprintf (const char *format, ...) __attribute__ ((format (printf, 1, 2), weak));
|
|
|
|
void logprintf (const char *format, ...) {
|
|
|
|
va_list ap;
|
|
|
|
va_start (ap, format);
|
|
|
|
vfprintf (stdout, format, ap);
|
|
|
|
va_end (ap);
|
|
|
|
}
|
|
|
|
|
2014-01-10 20:54:47 +04:00
|
|
|
extern int verbosity;
|
2014-01-10 18:37:56 +04:00
|
|
|
|
2014-01-11 20:00:31 +04:00
|
|
|
long long total_allocated_bytes;
|
|
|
|
|
2014-01-10 15:32:57 +04:00
|
|
|
static void out_of_memory (void) {
|
2014-01-13 16:36:29 +04:00
|
|
|
fprintf (stderr, "Out of memory\n");
|
|
|
|
exit (1);
|
2014-01-10 15:32:57 +04:00
|
|
|
}
|
|
|
|
|
2014-01-10 18:37:56 +04:00
|
|
|
int tsnprintf (char *buf, int len, const char *format, ...) {
|
|
|
|
va_list ap;
|
|
|
|
va_start (ap, format);
|
|
|
|
int r = vsnprintf (buf, len, format, ap);
|
|
|
|
va_end (ap);
|
|
|
|
assert (r <= len && "tsnprintf buffer overflow");
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
int tasprintf (char **res, const char *format, ...) {
|
|
|
|
va_list ap;
|
|
|
|
va_start (ap, format);
|
|
|
|
int r = vasprintf (res, format, ap);
|
|
|
|
assert (r >= 0);
|
|
|
|
va_end (ap);
|
|
|
|
void *rs = talloc (strlen (*res) + 1);
|
|
|
|
memcpy (rs, *res, strlen (*res) + 1);
|
|
|
|
free (*res);
|
|
|
|
*res = rs;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2014-01-11 20:00:31 +04:00
|
|
|
void print_backtrace (void);
|
2014-01-10 21:13:27 +04:00
|
|
|
void tfree (void *ptr, int size __attribute__ ((unused))) {
|
2014-01-10 18:37:56 +04:00
|
|
|
#ifdef DEBUG
|
2014-01-11 20:00:31 +04:00
|
|
|
total_allocated_bytes -= size;
|
|
|
|
ptr -= RES_PRE;
|
2014-01-12 04:43:29 +04:00
|
|
|
if (size != (int)((*(int *)ptr) ^ 0xbedabeda)) {
|
|
|
|
logprintf ("size = %d, ptr = %d\n", size, (*(int *)ptr) ^ 0xbedabeda);
|
|
|
|
}
|
2014-01-10 18:37:56 +04:00
|
|
|
assert (*(int *)ptr == (int)((size) ^ 0xbedabeda));
|
2014-01-11 20:00:31 +04:00
|
|
|
assert (*(int *)(ptr + RES_PRE + size) == (int)((size) ^ 0x7bed7bed));
|
|
|
|
assert (*(int *)(ptr + 4) == size);
|
|
|
|
int block_num = *(int *)(ptr + 4 + RES_PRE + size);
|
|
|
|
if (block_num >= used_blocks) {
|
|
|
|
logprintf ("block_num = %d, used = %d\n", block_num, used_blocks);
|
|
|
|
}
|
|
|
|
assert (block_num < used_blocks);
|
|
|
|
if (block_num < used_blocks - 1) {
|
|
|
|
void *p = blocks[used_blocks - 1];
|
|
|
|
int s = (*(int *)p) ^ 0xbedabeda;
|
|
|
|
*(int *)(p + 4 + RES_PRE + s) = block_num;
|
|
|
|
blocks[block_num] = p;
|
|
|
|
}
|
|
|
|
blocks[--used_blocks] = 0;
|
|
|
|
memset (ptr, 0, size + RES_PRE + RES_AFTER);
|
|
|
|
*(int *)ptr = size + 12;
|
|
|
|
free_blocks[free_blocks_cnt ++] = ptr;
|
|
|
|
#else
|
2014-01-10 18:37:56 +04:00
|
|
|
free (ptr);
|
2014-01-11 20:00:31 +04:00
|
|
|
#endif
|
2014-01-10 18:37:56 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void tfree_str (void *ptr) {
|
|
|
|
if (!ptr) { return; }
|
|
|
|
tfree (ptr, strlen (ptr) + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void tfree_secure (void *ptr, int size) {
|
|
|
|
memset (ptr, 0, size);
|
|
|
|
tfree (ptr, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
void *trealloc (void *ptr, size_t old_size __attribute__ ((unused)), size_t size) {
|
|
|
|
#ifdef DEBUG
|
|
|
|
void *p = talloc (size);
|
|
|
|
memcpy (p, ptr, size >= old_size ? old_size : size);
|
|
|
|
tfree (ptr, old_size);
|
|
|
|
return p;
|
|
|
|
#else
|
2014-01-10 17:08:12 +04:00
|
|
|
void *p = realloc (ptr, size);
|
|
|
|
ensure_ptr (p);
|
|
|
|
return p;
|
2014-01-10 18:37:56 +04:00
|
|
|
#endif
|
2014-01-10 17:08:12 +04:00
|
|
|
}
|
|
|
|
|
2014-01-10 15:32:57 +04:00
|
|
|
void *talloc (size_t size) {
|
2014-01-10 18:37:56 +04:00
|
|
|
#ifdef DEBUG
|
2014-01-11 20:00:31 +04:00
|
|
|
total_allocated_bytes += size;
|
|
|
|
void *p = malloc (size + RES_PRE + RES_AFTER);
|
2014-01-10 18:37:56 +04:00
|
|
|
ensure_ptr (p);
|
|
|
|
*(int *)p = size ^ 0xbedabeda;
|
2014-01-11 20:00:31 +04:00
|
|
|
*(int *)(p + 4) = size;
|
|
|
|
*(int *)(p + RES_PRE + size) = size ^ 0x7bed7bed;
|
|
|
|
*(int *)(p + RES_AFTER + 4 + size) = used_blocks;
|
|
|
|
blocks[used_blocks ++] = p;
|
2014-08-12 16:20:00 +04:00
|
|
|
|
2014-08-12 19:15:04 +04:00
|
|
|
if (used_blocks - 1 == 24867) {
|
|
|
|
assert (0);
|
|
|
|
}
|
2014-08-12 16:20:00 +04:00
|
|
|
tcheck ();
|
2014-01-11 20:00:31 +04:00
|
|
|
return p + 8;
|
2014-01-10 18:37:56 +04:00
|
|
|
#else
|
2014-01-10 15:32:57 +04:00
|
|
|
void *p = malloc (size);
|
|
|
|
ensure_ptr (p);
|
|
|
|
return p;
|
2014-01-10 18:37:56 +04:00
|
|
|
#endif
|
2014-01-10 15:32:57 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void *talloc0 (size_t size) {
|
|
|
|
void *p = talloc (size);
|
|
|
|
memset (p, 0, size);
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *tstrdup (const char *s) {
|
2014-01-10 18:37:56 +04:00
|
|
|
#ifdef DEBUG
|
|
|
|
int l = strlen (s);
|
|
|
|
char *p = talloc (l + 1);
|
|
|
|
memcpy (p, s, l + 1);
|
|
|
|
return p;
|
|
|
|
#else
|
2014-01-10 15:32:57 +04:00
|
|
|
char *p = strdup (s);
|
|
|
|
if (p == NULL) {
|
|
|
|
out_of_memory ();
|
|
|
|
}
|
|
|
|
return p;
|
2014-01-10 18:37:56 +04:00
|
|
|
#endif
|
2014-01-10 15:32:57 +04:00
|
|
|
}
|
|
|
|
|
2014-01-13 16:26:48 +04:00
|
|
|
char *tstrndup (const char *s, size_t n) {
|
|
|
|
#ifdef DEBUG
|
|
|
|
size_t l = 0;
|
|
|
|
for (l = 0; l < n && s[l]; l++) { }
|
|
|
|
char *p = talloc (l + 1);
|
|
|
|
memcpy (p, s, l);
|
|
|
|
p[l] = 0;
|
|
|
|
return p;
|
|
|
|
#else
|
|
|
|
char *p = strndup (s, n);
|
|
|
|
if (p == NULL) {
|
|
|
|
out_of_memory ();
|
|
|
|
}
|
|
|
|
return p;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-01-10 15:32:57 +04:00
|
|
|
void ensure (int r) {
|
|
|
|
if (!r) {
|
|
|
|
logprintf ("Open SSL error\n");
|
|
|
|
ERR_print_errors_fp (stderr);
|
|
|
|
assert (0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ensure_ptr (void *p) {
|
|
|
|
if (p == NULL) {
|
|
|
|
out_of_memory ();
|
|
|
|
}
|
|
|
|
}
|
2014-01-10 20:54:47 +04:00
|
|
|
|
|
|
|
int tinflate (void *input, int ilen, void *output, int olen) {
|
|
|
|
z_stream strm;
|
|
|
|
memset (&strm, 0, sizeof (strm));
|
|
|
|
assert (inflateInit2 (&strm, 16 + MAX_WBITS) == Z_OK);
|
|
|
|
strm.avail_in = ilen;
|
|
|
|
strm.next_in = input;
|
|
|
|
strm.avail_out = olen ;
|
|
|
|
strm.next_out = output;
|
|
|
|
int err = inflate (&strm, Z_FINISH), total_out = 0;
|
|
|
|
if (err == Z_OK || err == Z_STREAM_END) {
|
|
|
|
total_out = (int) strm.total_out;
|
|
|
|
if (err == Z_STREAM_END && verbosity >= 2) {
|
|
|
|
logprintf ( "inflated %d bytes\n", (int) strm.total_out);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (verbosity && err != Z_STREAM_END) {
|
|
|
|
logprintf ( "inflate error = %d\n", err);
|
|
|
|
logprintf ( "inflated %d bytes\n", (int) strm.total_out);
|
|
|
|
}
|
|
|
|
inflateEnd (&strm);
|
|
|
|
return total_out;
|
|
|
|
}
|
2014-01-11 20:00:31 +04:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
void tcheck (void) {
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < used_blocks; i++) {
|
|
|
|
void *ptr = blocks[i];
|
|
|
|
int size = (*(int *)ptr) ^ 0xbedabeda;
|
2014-08-12 16:20:00 +04:00
|
|
|
if (!(*(int *)(ptr + 4) == size) ||
|
|
|
|
!(*(int *)(ptr + RES_PRE + size) == (size ^ 0x7bed7bed)) ||
|
|
|
|
!(*(int *)(ptr + RES_PRE + 4 + size) == i)) {
|
|
|
|
logprintf ("Bad block at address %p (size %d, num %d)\n", ptr, size, i);
|
|
|
|
assert (0 && "Bad block");
|
|
|
|
}
|
2014-01-11 20:00:31 +04:00
|
|
|
}
|
|
|
|
for (i = 0; i < free_blocks_cnt; i++) {
|
|
|
|
void *ptr = free_blocks[i];
|
|
|
|
int l = *(int *)ptr;
|
|
|
|
int j = 0;
|
|
|
|
for (j = 0; j < l; j++) {
|
|
|
|
if (*(char *)(ptr + 4 + j)) {
|
|
|
|
hexdump (ptr + 8, ptr + 8 + l + ((-l) & 3));
|
|
|
|
logprintf ("Used freed memory size = %d. ptr = %p\n", l + 4 - RES_PRE - RES_AFTER, ptr);
|
|
|
|
assert (0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-08-12 16:20:00 +04:00
|
|
|
//logprintf ("ok. Used_blocks = %d. Free blocks = %d\n", used_blocks, free_blocks_cnt);
|
2014-01-11 20:00:31 +04:00
|
|
|
}
|
2014-01-28 16:40:35 +04:00
|
|
|
|
|
|
|
void texists (void *ptr, int size) {
|
|
|
|
ptr -= RES_PRE;
|
|
|
|
if (size != (int)((*(int *)ptr) ^ 0xbedabeda)) {
|
|
|
|
logprintf ("size = %d, ptr = %d\n", size, (*(int *)ptr) ^ 0xbedabeda);
|
|
|
|
}
|
|
|
|
assert (*(int *)ptr == (int)((size) ^ 0xbedabeda));
|
|
|
|
assert (*(int *)(ptr + RES_PRE + size) == (int)((size) ^ 0x7bed7bed));
|
|
|
|
assert (*(int *)(ptr + 4) == size);
|
|
|
|
int block_num = *(int *)(ptr + 4 + RES_PRE + size);
|
|
|
|
if (block_num >= used_blocks) {
|
|
|
|
logprintf ("block_num = %d, used = %d\n", block_num, used_blocks);
|
|
|
|
}
|
|
|
|
assert (block_num < used_blocks);
|
|
|
|
}
|
2014-01-11 20:00:31 +04:00
|
|
|
#endif
|
2014-08-14 22:03:33 +04:00
|
|
|
|
|
|
|
void my_clock_gettime (int clock_id, struct timespec *T) {
|
|
|
|
#ifdef __MACH__
|
|
|
|
// We are ignoring MONOTONIC and hope time doesn't go back too often
|
|
|
|
clock_serv_t cclock;
|
|
|
|
mach_timespec_t mts;
|
|
|
|
host_get_clock_service(mach_host_self(), CALENDAR_CLOCK, &cclock);
|
|
|
|
clock_get_time(cclock, &mts);
|
|
|
|
mach_port_deallocate(mach_task_self(), cclock);
|
|
|
|
T->tv_sec = mts.tv_sec;
|
|
|
|
T->tv_nsec = mts.tv_nsec;
|
|
|
|
#else
|
|
|
|
assert (clock_gettime(clock_id, T) >= 0);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
double get_double_time (void) {
|
|
|
|
struct timespec tv;
|
|
|
|
my_clock_gettime (CLOCK_REALTIME, &tv);
|
|
|
|
return tv.tv_sec + 1e-9 * tv.tv_nsec;
|
|
|
|
}
|
|
|
|
|
|
|
|
void tglt_secure_random (void *s, int l) {
|
|
|
|
if (RAND_bytes (s, l) < 0) {
|
|
|
|
/*if (allow_weak_random) {
|
|
|
|
RAND_pseudo_bytes (s, l);
|
|
|
|
} else {*/
|
|
|
|
assert (0 && "End of random. If you want, you can start with -w");
|
|
|
|
//}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|