ddnet/src/engine/e_packer.c
2008-03-18 01:30:47 +00:00

214 lines
3.2 KiB
C

/* copyright (c) 2007 magnus auvinen, see licence.txt for more info */
#include <stdlib.h>
#include "e_system.h"
#include "e_packer.h"
#include "e_compression.h"
#include "e_engine.h"
#include "e_config.h"
/* useful for debugging */
#if 0
#define packing_error(p) p->error = 1; dbg_break()
#else
#define packing_error(p) p->error = 1
#endif
static int stress_get_int()
{
static const int nasty[] = {-1, 0, 1, 66000, -66000, (-1<<31), 0x7fffffff};
if(rand()&1)
return rand();
return nasty[rand()%6];
}
static const char *stress_get_string(int *size)
{
static char noise[1024];
int i;
int s;
s = (rand()%1024)-1;
for(i = 0; i < s; i++)
noise[i] = (rand()%254)+1;
noise[s] = 0;
if(size)
*size = s;
return noise;
}
static int stress_prob(float probability)
{
if(!config.dbg_stress_network)
return 0;
if(rand()/(float)RAND_MAX < probability)
return 1;
return 0;
}
void packer_reset(PACKER *p)
{
p->error = 0;
p->current = p->buffer;
p->end = p->current + PACKER_BUFFER_SIZE;
}
void packer_add_int(PACKER *p, int i)
{
if(p->error)
return;
if(stress_prob(0.025f))
i = stress_get_int();
/* make sure that we have space enough */
if(p->end - p->current < 6)
{
dbg_break();
p->error = 1;
}
else
p->current = vint_pack(p->current, i);
}
void packer_add_string(PACKER *p, const char *str, int limit)
{
if(p->error)
return;
if(stress_prob(0.1f))
{
str = stress_get_string(0);
limit = 0;
}
/* */
if(limit > 0)
{
while(*str && limit != 0)
{
*p->current++ = *str++;
limit--;
if(p->current >= p->end)
{
packing_error(p);
break;
}
}
*p->current++ = 0;
}
else
{
while(*str)
{
*p->current++ = *str++;
if(p->current >= p->end)
{
packing_error(p);
break;
}
}
*p->current++ = 0;
}
}
void packer_add_raw(PACKER *p, const unsigned char *data, int size)
{
if(p->error)
return;
if(p->current+size >= p->end)
{
packing_error(p);
return;
}
while(size)
{
*p->current++ = *data++;
size--;
}
}
int packer_size(PACKER *p)
{
return (const unsigned char *)p->current-(const unsigned char *)p->buffer;
}
const unsigned char *packer_data(PACKER *p)
{
return (const unsigned char *)p->buffer;
}
void unpacker_reset(UNPACKER *p, const unsigned char *data, int size)
{
p->error = 0;
p->start = data;
p->end = p->start + size;
p->current = p->start;
}
int unpacker_get_int(UNPACKER *p)
{
int i;
if(p->error)
return 0;
if(p->current >= p->end)
{
packing_error(p);
return 0;
}
p->current = vint_unpack(p->current, &i);
if(p->current > p->end)
{
packing_error(p);
return 0;
}
return i;
}
const char *unpacker_get_string(UNPACKER *p)
{
char *ptr;
if(p->error || p->current >= p->end)
return "";
ptr = (char *)p->current;
while(*p->current) /* skip the string */
{
p->current++;
if(p->current == p->end)
{
packing_error(p);
return "";
}
}
p->current++;
/* sanitize all strings */
str_sanitize(ptr);
return ptr;
}
const unsigned char *unpacker_get_raw(UNPACKER *p, int size)
{
const unsigned char *ptr = p->current;
if(p->error)
return 0;
/* check for nasty sizes */
if(size < 0 || p->current+size > p->end)
{
packing_error(p);
return 0;
}
/* "unpack" the data */
p->current += size;
return ptr;
}