blob: 13be3234226da5c0a62e4fa092127fa214a3c7c9 [file] [log] [blame]
/*
* OpenVPN -- An application to securely tunnel IP networks
* over a single TCP/UDP port, with support for SSL/TLS-based
* session authentication and key exchange,
* packet encryption, packet authentication, and
* packet compression.
*
* Copyright (C) 2002-2018 OpenVPN Inc <sales@openvpn.net>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2
* as published by the Free Software Foundation.
*
* This program 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 program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#elif defined(_MSC_VER)
#include "config-msvc.h"
#endif
#include "syshead.h"
#include "buffer.h"
#include "misc.h"
#include "crypto.h"
#include "schedule.h"
#include "memdbg.h"
#ifdef SCHEDULE_TEST
struct status
{
int sru;
int ins;
int coll;
int lsteps;
};
static struct status z;
#endif
#ifdef ENABLE_DEBUG
static void
schedule_entry_debug_info(const char *caller, const struct schedule_entry *e)
{
struct gc_arena gc = gc_new();
if (e)
{
dmsg(D_SCHEDULER, "SCHEDULE: %s wakeup=[%s] pri=%u",
caller,
tv_string_abs(&e->tv, &gc),
e->pri);
}
else
{
dmsg(D_SCHEDULER, "SCHEDULE: %s NULL",
caller);
}
gc_free(&gc);
}
#endif
static inline void
schedule_set_pri(struct schedule_entry *e)
{
e->pri = random();
if (e->pri < 1)
{
e->pri = 1;
}
}
/* This is the master key comparison routine. A key is
* simply a struct timeval containing the absolute time for
* an event. The unique treap priority (pri) is used to ensure
* that keys do not collide.
*/
static inline int
schedule_entry_compare(const struct schedule_entry *e1,
const struct schedule_entry *e2)
{
if (e1->tv.tv_sec < e2->tv.tv_sec)
{
return -1;
}
else if (e1->tv.tv_sec > e2->tv.tv_sec)
{
return 1;
}
else
{
if (e1->tv.tv_usec < e2->tv.tv_usec)
{
return -1;
}
else if (e1->tv.tv_usec > e2->tv.tv_usec)
{
return 1;
}
else
{
if (e1->pri < e2->pri)
{
return -1;
}
else if (e1->pri > e2->pri)
{
return 1;
}
else
{
return 0;
}
}
}
}
/*
* Detach a btree node from its parent
*/
static inline void
schedule_detach_parent(struct schedule *s, struct schedule_entry *e)
{
if (e)
{
if (e->parent)
{
if (e->parent->lt == e)
{
e->parent->lt = NULL;
}
else if (e->parent->gt == e)
{
e->parent->gt = NULL;
}
else
{
/* parent <-> child linkage is corrupted */
ASSERT(0);
}
e->parent = NULL;
}
else
{
if (s->root == e) /* last element deleted, tree is empty */
{
s->root = NULL;
}
}
}
}
/*
*
* Given a binary search tree, move a node toward the root
* while still maintaining the correct ordering relationships
* within the tree. This function is the workhorse
* of the tree balancer.
*
* This code will break on key collisions, which shouldn't
* happen because the treap priority is considered part of the key
* and is guaranteed to be unique.
*/
static void
schedule_rotate_up(struct schedule *s, struct schedule_entry *e)
{
if (e && e->parent)
{
struct schedule_entry *lt = e->lt;
struct schedule_entry *gt = e->gt;
struct schedule_entry *p = e->parent;
struct schedule_entry *gp = p->parent;
if (gp) /* if grandparent exists, modify its child link */
{
if (gp->gt == p)
{
gp->gt = e;
}
else if (gp->lt == p)
{
gp->lt = e;
}
else
{
ASSERT(0);
}
}
else /* no grandparent, now we are the root */
{
s->root = e;
}
/* grandparent is now our parent */
e->parent = gp;
/* parent is now our child */
p->parent = e;
/* reorient former parent's links
* to reflect new position in the tree */
if (p->gt == e)
{
e->lt = p;
p->gt = lt;
if (lt)
{
lt->parent = p;
}
}
else if (p->lt == e)
{
e->gt = p;
p->lt = gt;
if (gt)
{
gt->parent = p;
}
}
else
{
/* parent <-> child linkage is corrupted */
ASSERT(0);
}
#ifdef SCHEDULE_TEST
++z.sru;
#endif
}
}
/*
* This is the treap deletion algorithm:
*
* Rotate lesser-priority children up in the tree
* until we are childless. Then delete.
*/
void
schedule_remove_node(struct schedule *s, struct schedule_entry *e)
{
while (e->lt || e->gt)
{
if (e->lt)
{
if (e->gt)
{
if (e->lt->pri < e->gt->pri)
{
schedule_rotate_up(s, e->lt);
}
else
{
schedule_rotate_up(s, e->gt);
}
}
else
{
schedule_rotate_up(s, e->lt);
}
}
else if (e->gt)
{
schedule_rotate_up(s, e->gt);
}
}
schedule_detach_parent(s, e);
e->pri = 0;
}
/*
* Trivially add a node to a binary search tree without
* regard for balance.
*/
static void
schedule_insert(struct schedule *s, struct schedule_entry *e)
{
struct schedule_entry *c = s->root;
while (true)
{
const int comp = schedule_entry_compare(e, c);
#ifdef SCHEDULE_TEST
++z.ins;
#endif
if (comp == -1)
{
if (c->lt)
{
c = c->lt;
continue;
}
else
{
c->lt = e;
e->parent = c;
break;
}
}
else if (comp == 1)
{
if (c->gt)
{
c = c->gt;
continue;
}
else
{
c->gt = e;
e->parent = c;
break;
}
}
else
{
/* rare key/priority collision -- no big deal,
* just choose another priority and retry */
#ifdef SCHEDULE_TEST
++z.coll;
#endif
schedule_set_pri(e);
/* msg (M_INFO, "PRI COLLISION pri=%u", e->pri); */
c = s->root;
continue;
}
}
}
/*
* Given an element, remove it from the btree if it's already
* there and re-insert it based on its current key.
*/
void
schedule_add_modify(struct schedule *s, struct schedule_entry *e)
{
#ifdef ENABLE_DEBUG
if (check_debug_level(D_SCHEDULER))
{
schedule_entry_debug_info("schedule_add_modify", e);
}
#endif
/* already in tree, remove */
if (IN_TREE(e))
{
schedule_remove_node(s, e);
}
/* set random priority */
schedule_set_pri(e);
if (s->root)
{
schedule_insert(s, e); /* trivial insert into tree */
}
else
{
s->root = e; /* tree was empty, we are the first element */
}
/* This is the magic of the randomized treap algorithm which
* keeps the tree balanced. Move the node up the tree until
* its own priority is greater than that of its parent */
while (e->parent && e->parent->pri > e->pri)
{
schedule_rotate_up(s, e);
}
}
/*
* Find the earliest event to be scheduled
*/
struct schedule_entry *
schedule_find_least(struct schedule_entry *e)
{
if (e)
{
while (e->lt)
{
#ifdef SCHEDULE_TEST
++z.lsteps;
#endif
e = e->lt;
}
}
#ifdef ENABLE_DEBUG
if (check_debug_level(D_SCHEDULER))
{
schedule_entry_debug_info("schedule_find_least", e);
}
#endif
return e;
}
/*
* Public functions below this point
*/
struct schedule *
schedule_init(void)
{
struct schedule *s;
ALLOC_OBJ_CLEAR(s, struct schedule);
return s;
}
void
schedule_free(struct schedule *s)
{
free(s);
}
void
schedule_remove_entry(struct schedule *s, struct schedule_entry *e)
{
s->earliest_wakeup = NULL; /* invalidate cache */
schedule_remove_node(s, e);
}
/*
* Debug functions below this point
*/
#ifdef SCHEDULE_TEST
static inline struct schedule_entry *
schedule_find_earliest_wakeup(struct schedule *s)
{
return schedule_find_least(s->root);
}
/*
* Recursively check that the treap (btree) is
* internally consistent.
*/
int
schedule_debug_entry(const struct schedule_entry *e,
int depth,
int *count,
struct timeval *least,
const struct timeval *min,
const struct timeval *max)
{
struct gc_arena gc = gc_new();
int maxdepth = depth;
if (e)
{
int d;
ASSERT(e != e->lt);
ASSERT(e != e->gt);
ASSERT(e != e->parent);
ASSERT(!e->parent || e->parent != e->lt);
ASSERT(!e->parent || e->parent != e->gt);
ASSERT(!e->lt || e->lt != e->gt);
if (e->lt)
{
ASSERT(e->lt->parent == e);
ASSERT(schedule_entry_compare(e->lt, e) == -1);
ASSERT(e->lt->pri >= e->pri);
}
if (e->gt)
{
ASSERT(e->gt->parent == e);
ASSERT(schedule_entry_compare(e->gt, e));
ASSERT(e->gt->pri >= e->pri);
}
ASSERT(tv_le(min, &e->tv));
ASSERT(tv_le(&e->tv, max));
if (count)
{
++(*count);
}
if (least && tv_lt(&e->tv, least))
{
*least = e->tv;
}
d = schedule_debug_entry(e->lt, depth+1, count, least, min, &e->tv);
if (d > maxdepth)
{
maxdepth = d;
}
d = schedule_debug_entry(e->gt, depth+1, count, least, &e->tv, max);
if (d > maxdepth)
{
maxdepth = d;
}
}
gc_free(&gc);
return maxdepth;
}
int
schedule_debug(struct schedule *s, int *count, struct timeval *least)
{
struct timeval min;
struct timeval max;
min.tv_sec = 0;
min.tv_usec = 0;
max.tv_sec = 0x7FFFFFFF;
max.tv_usec = 0x7FFFFFFF;
if (s->root)
{
ASSERT(s->root->parent == NULL);
}
return schedule_debug_entry(s->root, 0, count, least, &min, &max);
}
#if 1
void
tv_randomize(struct timeval *tv)
{
tv->tv_sec += random() % 100;
tv->tv_usec = random() % 100;
}
#else /* if 1 */
void
tv_randomize(struct timeval *tv)
{
struct gc_arena gc = gc_new();
long int choice = get_random();
if ((choice & 0xFF) == 0)
{
tv->tv_usec += ((choice >> 8) & 0xFF);
}
else
{
prng_bytes((uint8_t *)tv, sizeof(struct timeval));
}
gc_free(&gc);
}
#endif /* if 1 */
void
schedule_verify(struct schedule *s)
{
struct gc_arena gc = gc_new();
struct timeval least;
int count;
int maxlev;
struct schedule_entry *e;
const struct status zz = z;
least.tv_sec = least.tv_usec = 0x7FFFFFFF;
count = 0;
maxlev = schedule_debug(s, &count, &least);
e = schedule_find_earliest_wakeup(s);
if (e)
{
printf("Verification Phase count=%d maxlev=%d sru=%d ins=%d coll=%d ls=%d l=%s",
count,
maxlev,
zz.sru,
zz.ins,
zz.coll,
zz.lsteps,
tv_string(&e->tv, &gc));
if (!tv_eq(&least, &e->tv))
{
printf(" [COMPUTED DIFFERENT MIN VALUES!]");
}
printf("\n");
}
CLEAR(z);
gc_free(&gc);
}
void
schedule_randomize_array(struct schedule_entry **array, int size)
{
int i;
for (i = 0; i < size; ++i)
{
const int src = get_random() % size;
struct schedule_entry *tmp = array [i];
if (i != src)
{
array [i] = array [src];
array [src] = tmp;
}
}
}
void
schedule_print_work(struct schedule_entry *e, int indent)
{
struct gc_arena gc = gc_new();
int i;
for (i = 0; i < indent; ++i)
{
printf(" ");
}
if (e)
{
printf("%s [%u] e=" ptr_format ", p=" ptr_format " lt=" ptr_format " gt=" ptr_format "\n",
tv_string(&e->tv, &gc),
e->pri,
(ptr_type)e,
(ptr_type)e->parent,
(ptr_type)e->lt,
(ptr_type)e->gt);
schedule_print_work(e->lt, indent+1);
schedule_print_work(e->gt, indent+1);
}
else
{
printf("NULL\n");
}
gc_free(&gc);
}
void
schedule_print(struct schedule *s)
{
printf("*************************\n");
schedule_print_work(s->root, 0);
}
void
schedule_test(void)
{
struct gc_arena gc = gc_new();
int n = 1000;
int n_mod = 25;
int i, j;
struct schedule_entry **array;
struct schedule *s = schedule_init();
struct schedule_entry *e;
CLEAR(z);
ALLOC_ARRAY(array, struct schedule_entry *, n);
printf("Creation/Insertion Phase\n");
for (i = 0; i < n; ++i)
{
ALLOC_OBJ_CLEAR(array[i], struct schedule_entry);
tv_randomize(&array[i]->tv);
/*schedule_print (s);*/
/*schedule_verify (s);*/
schedule_add_modify(s, array[i]);
}
schedule_randomize_array(array, n);
/*schedule_print (s);*/
schedule_verify(s);
for (j = 1; j <= n_mod; ++j)
{
printf("Modification Phase Pass %d\n", j);
for (i = 0; i < n; ++i)
{
e = schedule_find_earliest_wakeup(s);
/*printf ("BEFORE %s\n", tv_string (&e->tv, &gc));*/
tv_randomize(&e->tv);
/*printf ("AFTER %s\n", tv_string (&e->tv, &gc));*/
schedule_add_modify(s, e);
/*schedule_verify (s);*/
/*schedule_print (s);*/
}
schedule_verify(s);
/*schedule_print (s);*/
}
/*printf ("INS=%d\n", z.ins);*/
while ((e = schedule_find_earliest_wakeup(s)))
{
schedule_remove_node(s, e);
/*schedule_verify (s);*/
}
schedule_verify(s);
printf("S->ROOT is %s\n", s->root ? "NOT NULL" : "NULL");
for (i = 0; i < n; ++i)
{
free(array[i]);
}
free(array);
free(s);
gc_free(&gc);
}
#endif /* ifdef SCHEDULE_TEST */