| /* |
| * qrencode - QR Code encoder |
| * |
| * Masking. |
| * 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 <stdlib.h> |
| #include <string.h> |
| #include <limits.h> |
| #include <errno.h> |
| |
| #include "qrencode.h" |
| #include "qrspec.h" |
| #include "mask.h" |
| |
| static 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(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(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 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 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 int Mask_calcRunLength(int width, unsigned char *frame, int dir, int *runLength) |
| { |
| int head; |
| int i; |
| unsigned char *p; |
| int pitch; |
| |
| pitch = (dir==0)?1:width; |
| if(frame[0] & 1) { |
| runLength[0] = -1; |
| head = 1; |
| } else { |
| head = 0; |
| } |
| runLength[head] = 1; |
| p = frame + pitch; |
| |
| for(i=1; i<width; i++) { |
| if((p[0] ^ p[-pitch]) & 1) { |
| head++; |
| runLength[head] = 1; |
| } else { |
| runLength[head]++; |
| } |
| p += pitch; |
| } |
| |
| return head + 1; |
| } |
| |
| static 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_calcRunLength(width, frame + y * width, 0, runLength); |
| demerit += Mask_calcN1N3(length, runLength); |
| } |
| |
| for(x=0; x<width; x++) { |
| length = Mask_calcRunLength(width, frame + x, 1, 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(w2); |
| if(mask == NULL) return NULL; |
| bestMask = 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; |
| free(bestMask); |
| bestMask = mask; |
| mask = (unsigned char *)malloc(w2); |
| if(mask == NULL) break; |
| } |
| } |
| free(mask); |
| return bestMask; |
| } |