123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358 |
- /*
- * qrencode - QR Code encoder
- *
- * Masking.
- * Copyright (C) 2006-2017 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 <stdlib.h>
- #include <string.h>
- #include <limits.h>
- #include <errno.h>
- #include "qrencode.h"
- #include "qrspec.h"
- #include "mask.h"
- #define STATIC_IN_RELEASE
- STATIC_IN_RELEASE int Mask_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;
- }
- /**
- * Demerit coefficients.
- * See Section 8.8.2, pp.45, JIS X0510:2004.
- */
- #define N1 (3)
- #define N2 (3)
- #define N3 (40)
- #define N4 (10)
- #define MASKMAKER(__exp__) \
- int x, y;\
- int b = 0;\
- \
- for(y = 0; y < width; y++) {\
- for(x = 0; x < width; x++) {\
- if(*s & 0x80) {\
- *d = *s;\
- } else {\
- *d = *s ^ ((__exp__) == 0);\
- }\
- b += (int)(*d & 1);\
- s++; d++;\
- }\
- }\
- return b;
- static int Mask_mask0(int width, const unsigned char *s, unsigned char *d)
- {
- MASKMAKER((x+y)&1)
- }
- static int Mask_mask1(int width, const unsigned char *s, unsigned char *d)
- {
- MASKMAKER(y&1)
- }
- static int Mask_mask2(int width, const unsigned char *s, unsigned char *d)
- {
- MASKMAKER(x%3)
- }
- static int Mask_mask3(int width, const unsigned char *s, unsigned char *d)
- {
- MASKMAKER((x+y)%3)
- }
- static int Mask_mask4(int width, const unsigned char *s, unsigned char *d)
- {
- MASKMAKER(((y/2)+(x/3))&1)
- }
- static int Mask_mask5(int width, const unsigned char *s, unsigned char *d)
- {
- MASKMAKER(((x*y)&1)+(x*y)%3)
- }
- static int Mask_mask6(int width, const unsigned char *s, unsigned char *d)
- {
- MASKMAKER((((x*y)&1)+(x*y)%3)&1)
- }
- static int Mask_mask7(int width, const unsigned char *s, unsigned char *d)
- {
- MASKMAKER((((x*y)%3)+((x+y)&1))&1)
- }
- #define maskNum (8)
- typedef int MaskMaker(int, const unsigned char *, unsigned char *);
- static MaskMaker *maskMakers[maskNum] = {
- Mask_mask0, Mask_mask1, Mask_mask2, Mask_mask3,
- Mask_mask4, Mask_mask5, Mask_mask6, Mask_mask7
- };
- #ifdef WITH_TESTS
- unsigned char *Mask_makeMaskedFrame(int width, unsigned char *frame, int mask)
- {
- unsigned char *masked;
- masked = (unsigned char *)malloc((size_t)(width * width));
- if(masked == NULL) return NULL;
- maskMakers[mask](width, frame, masked);
- return masked;
- }
- #endif
- unsigned char *Mask_makeMask(int width, unsigned char *frame, int mask, QRecLevel level)
- {
- unsigned char *masked;
- if(mask < 0 || mask >= maskNum) {
- errno = EINVAL;
- return NULL;
- }
- masked = (unsigned char *)malloc((size_t)(width * width));
- if(masked == NULL) return NULL;
- maskMakers[mask](width, frame, masked);
- Mask_writeFormatInformation(width, masked, mask, level);
- return masked;
- }
- //static int n1;
- //static int n2;
- //static int n3;
- //static int n4;
- STATIC_IN_RELEASE int Mask_calcN1N3(int length, int *runLength)
- {
- int i;
- int demerit = 0;
- int fact;
- for(i = 0; i < length; i++) {
- if(runLength[i] >= 5) {
- demerit += N1 + (runLength[i] - 5);
- //n1 += N1 + (runLength[i] - 5);
- }
- if((i & 1)) {
- if(i >= 3 && i < length-2 && (runLength[i] % 3) == 0) {
- fact = runLength[i] / 3;
- if(runLength[i-2] == fact &&
- runLength[i-1] == fact &&
- runLength[i+1] == fact &&
- runLength[i+2] == fact) {
- if(i == 3 || runLength[i-3] >= 4 * fact) {
- demerit += N3;
- //n3 += N3;
- } else if(i+4 >= length || runLength[i+3] >= 4 * fact) {
- demerit += N3;
- //n3 += N3;
- }
- }
- }
- }
- }
- return demerit;
- }
- STATIC_IN_RELEASE int Mask_calcN2(int width, unsigned char *frame)
- {
- int x, y;
- unsigned char *p;
- unsigned char b22, w22;
- int demerit = 0;
- p = frame + width + 1;
- for(y = 1; y < width; y++) {
- for(x = 1; x < width; x++) {
- b22 = p[0] & p[-1] & p[-width] & p [-width-1];
- w22 = p[0] | p[-1] | p[-width] | p [-width-1];
- if((b22 | (w22 ^ 1))&1) {
- demerit += N2;
- }
- p++;
- }
- p++;
- }
- return demerit;
- }
- STATIC_IN_RELEASE int Mask_calcRunLengthH(int width, unsigned char *frame, int *runLength)
- {
- int head;
- int i;
- unsigned char prev;
- if(frame[0] & 1) {
- runLength[0] = -1;
- head = 1;
- } else {
- head = 0;
- }
- runLength[head] = 1;
- prev = frame[0];
- for(i = 1; i < width; i++) {
- if((frame[i] ^ prev) & 1) {
- head++;
- runLength[head] = 1;
- prev = frame[i];
- } else {
- runLength[head]++;
- }
- }
- return head + 1;
- }
- STATIC_IN_RELEASE int Mask_calcRunLengthV(int width, unsigned char *frame, int *runLength)
- {
- int head;
- int i;
- unsigned char prev;
- if(frame[0] & 1) {
- runLength[0] = -1;
- head = 1;
- } else {
- head = 0;
- }
- runLength[head] = 1;
- prev = frame[0];
- for(i = 1; i < width; i++) {
- if((frame[i * width] ^ prev) & 1) {
- head++;
- runLength[head] = 1;
- prev = frame[i * width];
- } else {
- runLength[head]++;
- }
- }
- return head + 1;
- }
- STATIC_IN_RELEASE int Mask_evaluateSymbol(int width, unsigned char *frame)
- {
- int x, y;
- int demerit = 0;
- int runLength[QRSPEC_WIDTH_MAX + 1];
- int length;
- demerit += Mask_calcN2(width, frame);
- for(y = 0; y < width; y++) {
- length = Mask_calcRunLengthH(width, frame + y * width, runLength);
- demerit += Mask_calcN1N3(length, runLength);
- }
- for(x = 0; x < width; x++) {
- length = Mask_calcRunLengthV(width, frame + x, runLength);
- demerit += Mask_calcN1N3(length, runLength);
- }
- return demerit;
- }
- unsigned char *Mask_mask(int width, unsigned char *frame, QRecLevel level)
- {
- int i;
- unsigned char *mask, *bestMask;
- int minDemerit = INT_MAX;
- int blacks;
- int bratio;
- int demerit;
- int w2 = width * width;
- mask = (unsigned char *)malloc((size_t)w2);
- if(mask == NULL) return NULL;
- bestMask = (unsigned char *)malloc((size_t)w2);
- if(bestMask == NULL) {
- free(mask);
- return NULL;
- }
- for(i = 0; i < maskNum; i++) {
- // n1 = n2 = n3 = n4 = 0;
- demerit = 0;
- blacks = maskMakers[i](width, frame, mask);
- blacks += Mask_writeFormatInformation(width, mask, i, level);
- bratio = (200 * blacks + w2) / w2 / 2; /* (int)(100*blacks/w2+0.5) */
- demerit = (abs(bratio - 50) / 5) * N4;
- // n4 = demerit;
- demerit += Mask_evaluateSymbol(width, mask);
- // printf("(%d,%d,%d,%d)=%d\n", n1, n2, n3 ,n4, demerit);
- if(demerit < minDemerit) {
- minDemerit = demerit;
- memcpy(bestMask, mask, (size_t)w2);
- }
- }
- free(mask);
- return bestMask;
- }
|