summaryrefslogtreecommitdiff
path: root/protocols/Telegram/src/tgl/mtproto-utils.c
diff options
context:
space:
mode:
Diffstat (limited to 'protocols/Telegram/src/tgl/mtproto-utils.c')
-rw-r--r--protocols/Telegram/src/tgl/mtproto-utils.c178
1 files changed, 178 insertions, 0 deletions
diff --git a/protocols/Telegram/src/tgl/mtproto-utils.c b/protocols/Telegram/src/tgl/mtproto-utils.c
new file mode 100644
index 0000000000..481bff6371
--- /dev/null
+++ b/protocols/Telegram/src/tgl/mtproto-utils.c
@@ -0,0 +1,178 @@
+#include "config.h"
+#include <openssl/bn.h>
+#include "tgl.h"
+#include "tools.h"
+
+#if !defined(HAVE___BUILTIN_BSWAP32) && !defined(__FreeBSD__) && !defined(__OpenBSD__)
+static inline unsigned __builtin_bswap32(unsigned x) {
+ return ((x << 24) & 0xff000000) |
+ ((x << 8) & 0x00ff0000) |
+ ((x >> 8) & 0x0000ff00) |
+ ((x >> 24) & 0x000000ff);
+}
+#endif
+
+static unsigned long long gcd (unsigned long long a, unsigned long long b) {
+ return b ? gcd (b, a % b) : a;
+}
+
+static int check_prime (struct tgl_state *TLS, BIGNUM *p) {
+ int r = BN_is_prime (p, BN_prime_checks, 0, TLS->BN_ctx, 0);
+ ensure (r >= 0);
+ return r;
+}
+
+
+// Complete set of checks see at https://core.telegram.org/mtproto/security_guidelines
+
+
+// Checks that (p,g) is acceptable pair for DH
+int tglmp_check_DH_params (struct tgl_state *TLS, BIGNUM *p, int g) {
+ if (g < 2 || g > 7) { return -1; }
+ if (BN_num_bits (p) != 2048) { return -1; }
+
+ BIGNUM *t = BN_new ();
+
+ BIGNUM *dh_g = BN_new ();
+
+ ensure (BN_set_word (dh_g, 4 * g));
+ ensure (BN_mod (t, p, dh_g, TLS->BN_ctx));
+ int x = BN_get_word (t);
+ assert (x >= 0 && x < 4 * g);
+
+ BN_free (dh_g);
+
+ int res = 0;
+ switch (g) {
+ case 2:
+ if (x != 7) { res = -1; }
+ break;
+ case 3:
+ if (x % 3 != 2) { res = -1; }
+ break;
+ case 4:
+ break;
+ case 5:
+ if (x % 5 != 1 && x % 5 != 4) { res = -1; }
+ break;
+ case 6:
+ if (x != 19 && x != 23) { res = -1; }
+ break;
+ case 7:
+ if (x % 7 != 3 && x % 7 != 5 && x % 7 != 6) { res = -1; }
+ break;
+ }
+
+ if (res < 0 || !check_prime (TLS, p)) {
+ BN_free (t);
+ return -1;
+ }
+
+ BIGNUM *b = BN_new ();
+ ensure (BN_set_word (b, 2));
+ ensure (BN_div (t, 0, p, b, TLS->BN_ctx));
+ if (!check_prime (TLS, t)) {
+ res = -1;
+ }
+ BN_free (b);
+ BN_free (t);
+ return res;
+}
+
+// checks that g_a is acceptable for DH
+int tglmp_check_g_a (struct tgl_state *TLS, BIGNUM *p, BIGNUM *g_a) {
+ if (BN_num_bytes (g_a) > 256) {
+ return -1;
+ }
+ if (BN_num_bits (g_a) < 2048 - 64) {
+ return -1;
+ }
+ if (BN_cmp (p, g_a) <= 0) {
+ return -1;
+ }
+
+ BIGNUM *dif = BN_new ();
+ BN_sub (dif, p, g_a);
+ if (BN_num_bits (dif) < 2048 - 64) {
+ BN_free (dif);
+ return -1;
+ }
+ BN_free (dif);
+ return 0;
+}
+
+static unsigned long long BN2ull (BIGNUM *b) {
+ if (sizeof (BN_ULONG) == 8) {
+ return BN_get_word (b);
+ } else {
+ unsigned int tmp[2];
+ memset (tmp, 0, 8);
+ BN_bn2bin (b, (void *)tmp);
+ return __builtin_bswap32 (tmp[0]) * (1ll << 32) | __builtin_bswap32 (tmp[1]);
+ }
+}
+
+static void ull2BN (BIGNUM *b, unsigned long long val) {
+ if (sizeof (BN_ULONG) == 8 || val < (1ll << 32)) {
+ BN_set_word (b, val);
+ } else {
+ unsigned int tmp[2];
+ tmp[0] = __builtin_bswap32 (val >> 32);
+ tmp[1] = __builtin_bswap32 ((unsigned)val);
+ BN_bin2bn ((void *)tmp, 8, b);
+ }
+}
+
+int bn_factorize (BIGNUM *pq, BIGNUM *p, BIGNUM *q) {
+ // Should work in any case
+ // Rewrite this code
+ unsigned long long what = BN2ull (pq);
+
+ int it = 0;
+
+ unsigned long long g = 0;
+ int i;
+ for (i = 0; i < 3 || it < 1000; i++) {
+ int q = ((lrand48() & 15) + 17) % what;
+ unsigned long long x = (long long)lrand48 () % (what - 1) + 1, y = x;
+ int lim = 1 << (i + 18);
+ int j;
+ for (j = 1; j < lim; j++) {
+ ++it;
+ unsigned long long a = x, b = x, c = q;
+ while (b) {
+ if (b & 1) {
+ c += a;
+ if (c >= what) {
+ c -= what;
+ }
+ }
+ a += a;
+ if (a >= what) {
+ a -= what;
+ }
+ b >>= 1;
+ }
+ x = c;
+ unsigned long long z = x < y ? what + x - y : x - y;
+ g = gcd (z, what);
+ if (g != 1) {
+ break;
+ }
+ if (!(j & (j - 1))) {
+ y = x;
+ }
+ }
+ if (g > 1 && g < what) break;
+ }
+
+ assert (g > 1 && g < what);
+ unsigned long long p1 = g;
+ unsigned long long p2 = what / g;
+ if (p1 > p2) {
+ unsigned long long t = p1; p1 = p2; p2 = t;
+ }
+ ull2BN (p, p1);
+ ull2BN (q, p2);
+ return 0;
+}