⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 qrencode.c

📁 二维码QR的linux平台下的编码源代码
💻 C
字号:
/* * qrencode - QR Code encoder * * Copyright (C) 2006,2007 Kentaro Fukuchi <fukuchi@megaui.net> * * 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 */#include <stdio.h>#include <stdlib.h>#include <string.h>#include "qrencode.h"#include "qrencode_inner.h"#include "qrspec.h"#include "bitstream.h"#include "qrinput.h"#include "rscode.h"#include "split.h"#include "mask.h"/****************************************************************************** * Raw code *****************************************************************************/static void RSblock_init(RSblock *block, int dl, unsigned char *data, int el){	RS *rs;	block->dataLength = dl;	block->data = data;	block->eccLength = el;	block->ecc = (unsigned char *)malloc(el);	rs = init_rs(8, 0x11d, 0, 1, el, 255 - dl - el);	encode_rs_char(rs, data, block->ecc);}QRRawCode *QRraw_new(QRinput *input){	QRRawCode *raw;	int *spec;	int i;	RSblock *rsblock;	unsigned char *p;	p = QRinput_getByteStream(input);	if(p == NULL) {		return NULL;	}	raw = (QRRawCode *)malloc(sizeof(QRRawCode));	raw->datacode = p;	spec = QRspec_getEccSpec(input->version, input->level);	raw->version = input->version;	raw->blocks = QRspec_rsBlockNum(spec);	raw->rsblock = (RSblock *)malloc(sizeof(RSblock) * raw->blocks);	rsblock = raw->rsblock;	p = raw->datacode;	for(i=0; i<QRspec_rsBlockNum1(spec); i++) {		RSblock_init(rsblock, QRspec_rsDataCodes1(spec), p,						QRspec_rsEccCodes1(spec));		p += QRspec_rsDataCodes1(spec);		rsblock++;	}	for(i=0; i<QRspec_rsBlockNum2(spec); i++) {		RSblock_init(rsblock, QRspec_rsDataCodes2(spec), p,						QRspec_rsEccCodes2(spec));		p += QRspec_rsDataCodes2(spec);		rsblock++;	}	raw->b1 = QRspec_rsBlockNum1(spec);	raw->dataLength = QRspec_rsBlockNum1(spec) * QRspec_rsDataCodes1(spec)					+ QRspec_rsBlockNum2(spec) * QRspec_rsDataCodes2(spec);	raw->eccLength = QRspec_rsBlockNum(spec) * QRspec_rsEccCodes1(spec);	raw->count = 0;	free(spec);	return raw;}/** * Return a code (byte). * This function can be called iteratively. * @param raw raw code. * @return code */unsigned char QRraw_getCode(QRRawCode *raw){	int col, row;	unsigned char ret;	if(raw->count < raw->dataLength) {		row = raw->count % raw->blocks;		col = raw->count / raw->blocks;		if(col >= raw->rsblock[row].dataLength) {			row += raw->b1;		}		ret = raw->rsblock[row].data[col];	} else if(raw->count < raw->dataLength + raw->eccLength) {		row = (raw->count - raw->dataLength) % raw->blocks;		col = (raw->count - raw->dataLength) / raw->blocks;		ret = raw->rsblock[row].ecc[col];	} else {		return 0;	}	raw->count++;	return ret;}void QRraw_free(QRRawCode *raw){	int i;	free(raw->datacode);	for(i=0; i<raw->blocks; i++) {		free(raw->rsblock[i].ecc);	}	free(raw->rsblock);	free(raw);}/****************************************************************************** * Frame filling *****************************************************************************/typedef struct {	int width;	unsigned char *frame;	int x, y;	int dir;	int bit;} FrameFiller;static FrameFiller *FrameFiller_new(int width, unsigned char *frame){	FrameFiller *filler;	filler = (FrameFiller *)malloc(sizeof(FrameFiller));	filler->width = width;	filler->frame = frame;	filler->x = width - 1;	filler->y = width - 1;	filler->dir = -1;	filler->bit = -1;	return filler;}static unsigned char *FrameFiller_next(FrameFiller *filler){	unsigned char *p;	int x, y, w;	if(filler->bit == -1) {		filler->bit = 0;		return filler->frame + filler->y * filler->width + filler->x;	}	x = filler->x;	y = filler->y;	p = filler->frame;	w = filler->width;	if(filler->bit == 0) {		x--;		filler->bit++;	} else {		x++;		y += filler->dir;		filler->bit--;	}	if(filler->dir < 0) {		if(y < 0) {			y = 0;			x -= 2;			filler->dir = 1;			if(x == 6) {				x--;				y = 9;			}		}	} else {		if(y == w) {			y = w - 1;			x -= 2;			filler->dir = -1;			if(x == 6) {				x--;				y -= 8;			}		}	}	if(x < 0 || y < 0) return NULL;	filler->x = x;	filler->y = y;	if(p[y * w + x] & 0x80) {		// This tail recursion could be optimized.		return FrameFiller_next(filler);	}	return &p[y * w + x];}unsigned char *FrameFiller_fillerTest(int version){	int width, length;	unsigned char *frame, *p;	FrameFiller *filler;	int i, j;	unsigned char cl = 1;	unsigned char ch = 0;	width = QRspec_getWidth(version);	frame = QRspec_newFrame(version);	filler = FrameFiller_new(width, frame);	length = QRspec_getDataLength(version, QR_ECLEVEL_L)			+ QRspec_getECCLength(version, QR_ECLEVEL_L);	for(i=0; i<length; i++) {		for(j=0; j<8; j++) {			p = FrameFiller_next(filler);			*p = ch | cl;			cl++;			if(cl == 9) {				cl = 1;				ch += 0x10;			}		}	}	length = QRspec_getRemainder(version);	for(i=0; i<length; i++) {		p = FrameFiller_next(filler);		*p = 0xa0;	}	p = FrameFiller_next(filler);	free(filler);	if(p != NULL) {		return NULL;	}	return frame;}/****************************************************************************** * Format information *****************************************************************************/int QRcode_writeFormatInformation(int width, unsigned char *frame, int mask, QRecLevel level){	unsigned int format;	unsigned char v;	int i;	int blacks = 0;	format =  QRspec_getFormatInfo(mask, level);	for(i=0; i<8; i++) {		if(format & 1) {			blacks += 2;			v = 0x85;		} else {			v = 0x84;		}		frame[width * 8 + width - 1 - i] = v;		if(i < 6) {			frame[width * i + 8] = v;		} else {			frame[width * (i + 1) + 8] = v;		}		format= format >> 1;	}	for(i=0; i<7; i++) {		if(format & 1) {			blacks += 2;			v = 0x85;		} else {			v = 0x84;		}		frame[width * (width - 7 + i) + 8] = v;		if(i == 0) {			frame[width * 8 + 7] = v;		} else {			frame[width * 8 + 6 - i] = v;		}		format= format >> 1;	}	return blacks;}/****************************************************************************** * QR-code encoding *****************************************************************************/static QRcode *QRcode_new(int version, int width, unsigned char *data){	QRcode *qrcode;	qrcode = (QRcode *)malloc(sizeof(QRcode));	qrcode->version = version;	qrcode->width = width;	qrcode->data = data;	return qrcode;}void QRcode_free(QRcode *qrcode){	if(qrcode == NULL) return;	if(qrcode->data != NULL) {		free(qrcode->data);	}	free(qrcode);}QRcode *QRcode_encodeMask(QRinput *input, int version, QRecLevel level, int mask){	int width;	QRRawCode *raw;	unsigned char *frame, *masked, *p, code, bit;	FrameFiller *filler;	int i, j;	QRcode *qrcode;	QRinput_setVersion(input, version);	QRinput_setErrorCorrectionLevel(input, level);	raw = QRraw_new(input);	if(raw == NULL) {		return NULL;	}	version = raw->version;	width = QRspec_getWidth(version);	frame = QRspec_newFrame(version);	filler = FrameFiller_new(width, frame);	/* inteleaved data and ecc codes */	for(i=0; i<raw->dataLength + raw->eccLength; i++) {		code = QRraw_getCode(raw);		bit = 0x80;		for(j=0; j<8; j++) {			p = FrameFiller_next(filler);			*p = 0x02 | ((bit & code) != 0);			bit = bit >> 1;		}	}	QRraw_free(raw);	/* remainder bits */	j = QRspec_getRemainder(version);	for(i=0; i<j; i++) {		p = FrameFiller_next(filler);		*p = 0x02;	}	free(filler);	/* masking */	if(mask < 0) {		masked = Mask_mask(width, frame, level);	} else {		masked = Mask_makeMask(width, frame, mask);		QRcode_writeFormatInformation(width, masked, mask, QRinput_getErrorCorrectionLevel(input));	}	qrcode = QRcode_new(version, width, masked);	free(frame);	return qrcode;}QRcode *QRcode_encodeInput(QRinput *input, int version, QRecLevel level){	return QRcode_encodeMask(input, version, level, -1);}QRcode *QRcode_encodeString8bit(const char *string, int version, QRecLevel level){	QRinput *input;	QRcode *code;	input = QRinput_new();	QRinput_append(input, QR_MODE_8, strlen(string), (unsigned char *)string);	code = QRcode_encodeInput(input, version, level);	QRinput_free(input);	return code;}QRcode *QRcode_encodeString(const char *string, int version, QRecLevel level, QRencodeMode hint, int casesensitive){	QRinput *input;	QRcode *code;	if(hint != QR_MODE_8 && hint != QR_MODE_KANJI) {		return NULL;	}	input = QRinput_new();	Split_splitStringToQRinput(string, input, version, hint, casesensitive);	code = QRcode_encodeInput(input, version, level);	QRinput_free(input);	return code;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -