Não pode escolher mais do que 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.
 
 
 
 
 
 

238 linhas
4.2 KiB

/*
 * qrencode - QR Code encoder
 *
 * Binary sequence class.
 * Copyright (C) 2006-2011 Kentaro Fukuchi <kentaro@fukuchi.org>
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or any later version.
 *
 * This library 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
 */
#if HAVE_CONFIG_H
# include "config.h"
#endif
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "bitstream.h"
BitStream *BitStream_new(void)
{
	BitStream *bstream;
	bstream = (BitStream *)malloc(sizeof(BitStream));
	if(bstream == NULL) return NULL;
	bstream->length = 0;
	bstream->data = NULL;
	return bstream;
}
static int BitStream_allocate(BitStream *bstream, int length)
{
	unsigned char *data;
	if(bstream == NULL) {
		return -1;
	}
	data = (unsigned char *)malloc(length);
	if(data == NULL) {
		return -1;
	}
	if(bstream->data) {
		free(bstream->data);
	}
	bstream->length = length;
	bstream->data = data;
	return 0;
}
static BitStream *BitStream_newFromNum(int bits, unsigned int num)
{
	unsigned int mask;
	int i;
	unsigned char *p;
	BitStream *bstream;
	bstream = BitStream_new();
	if(bstream == NULL) return NULL;
	if(BitStream_allocate(bstream, bits)) {
		BitStream_free(bstream);
		return NULL;
	}
	p = bstream->data;
	mask = 1 << (bits - 1);
	for(i=0; i<bits; i++) {
		if(num & mask) {
			*p = 1;
		} else {
			*p = 0;
		}
		p++;
		mask = mask >> 1;
	}
	return bstream;
}
static BitStream *BitStream_newFromBytes(int size, unsigned char *data)
{
	unsigned char mask;
	int i, j;
	unsigned char *p;
	BitStream *bstream;
	bstream = BitStream_new();
	if(bstream == NULL) return NULL;
	if(BitStream_allocate(bstream, size * 8)) {
		BitStream_free(bstream);
		return NULL;
	}
	p = bstream->data;
	for(i=0; i<size; i++) {
		mask = 0x80;
		for(j=0; j<8; j++) {
			if(data[i] & mask) {
				*p = 1;
			} else {
				*p = 0;
			}
			p++;
			mask = mask >> 1;
		}
	}
	return bstream;
}
int BitStream_append(BitStream *bstream, BitStream *arg)
{
	unsigned char *data;
	if(arg == NULL) {
		return -1;
	}
	if(arg->length == 0) {
		return 0;
	}
	if(bstream->length == 0) {
		if(BitStream_allocate(bstream, arg->length)) {
			return -1;
		}
		memcpy(bstream->data, arg->data, arg->length);
		return 0;
	}
	data = (unsigned char *)malloc(bstream->length + arg->length);
	if(data == NULL) {
		return -1;
	}
	memcpy(data, bstream->data, bstream->length);
	memcpy(data + bstream->length, arg->data, arg->length);
	free(bstream->data);
	bstream->length += arg->length;
	bstream->data = data;
	return 0;
}
int BitStream_appendNum(BitStream *bstream, int bits, unsigned int num)
{
	BitStream *b;
	int ret;
	if(bits == 0) return 0;
	b = BitStream_newFromNum(bits, num);
	if(b == NULL) return -1;
	ret = BitStream_append(bstream, b);
	BitStream_free(b);
	return ret;
}
int BitStream_appendBytes(BitStream *bstream, int size, unsigned char *data)
{
	BitStream *b;
	int ret;
	if(size == 0) return 0;
	b = BitStream_newFromBytes(size, data);
	if(b == NULL) return -1;
	ret = BitStream_append(bstream, b);
	BitStream_free(b);
	return ret;
}
unsigned char *BitStream_toByte(BitStream *bstream)
{
	int i, j, size, bytes;
	unsigned char *data, v;
	unsigned char *p;
	size = BitStream_size(bstream);
	if(size == 0) {
		return NULL;
	}
	data = (unsigned char *)malloc((size + 7) / 8);
	if(data == NULL) {
		return NULL;
	}
	bytes = size  / 8;
	p = bstream->data;
	for(i=0; i<bytes; i++) {
		v = 0;
		for(j=0; j<8; j++) {
			v = v << 1;
			v |= *p;
			p++;
		}
		data[i] = v;
	}
	if(size & 7) {
		v = 0;
		for(j=0; j<(size & 7); j++) {
			v = v << 1;
			v |= *p;
			p++;
		}
		data[bytes] = v;
	}
	return data;
}
void BitStream_free(BitStream *bstream)
{
	if(bstream != NULL) {
		free(bstream->data);
		free(bstream);
	}
}