Merge pull request #2251 from yurydelendik/validate-charstrings
Validates type2 charstrings
This commit is contained in:
commit
9f460a9a8f
153
src/fonts.js
153
src/fonts.js
@ -5212,6 +5212,81 @@ var CFFFont = (function CFFFontClosure() {
|
|||||||
})();
|
})();
|
||||||
|
|
||||||
var CFFParser = (function CFFParserClosure() {
|
var CFFParser = (function CFFParserClosure() {
|
||||||
|
var CharstringValidationData = [
|
||||||
|
null,
|
||||||
|
{ id: 'hstem', min: 2, resetStack: true },
|
||||||
|
null,
|
||||||
|
{ id: 'vstem', min: 2, resetStack: true },
|
||||||
|
{ id: 'vmoveto', min: 1, resetStack: true },
|
||||||
|
{ id: 'rlineto', min: 2, resetStack: true },
|
||||||
|
{ id: 'hlineto', min: 1, resetStack: true },
|
||||||
|
{ id: 'vlineto', min: 1, resetStack: true },
|
||||||
|
{ id: 'rrcurveto', min: 6, resetStack: true },
|
||||||
|
null,
|
||||||
|
{ id: 'callsubr', min: 1, undefStack: true },
|
||||||
|
{ id: 'return', min: 0, resetStack: true },
|
||||||
|
null, // 12
|
||||||
|
null,
|
||||||
|
null, // endchar
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
{ id: 'hstemhm', min: 2, resetStack: true },
|
||||||
|
null, // hintmask
|
||||||
|
null, // cntrmask
|
||||||
|
{ id: 'rmoveto', min: 2, resetStack: true },
|
||||||
|
{ id: 'hmoveto', min: 1, resetStack: true },
|
||||||
|
{ id: 'vstemhm', min: 2, resetStack: true },
|
||||||
|
{ id: 'rcurveline', min: 8, resetStack: true },
|
||||||
|
{ id: 'rlinecurve', min: 8, resetStack: true },
|
||||||
|
{ id: 'vvcurveto', min: 4, resetStack: true },
|
||||||
|
{ id: 'hhcurveto', min: 4, resetStack: true },
|
||||||
|
null, // shortint
|
||||||
|
{ id: 'callgsubr', min: 1, undefStack: true },
|
||||||
|
{ id: 'vhcurveto', min: 4, resetStack: true },
|
||||||
|
{ id: 'hvcurveto', min: 4, resetStack: true }
|
||||||
|
];
|
||||||
|
var CharstringValidationData12 = [
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
{ id: 'and', min: 2, stackDelta: -1 },
|
||||||
|
{ id: 'or', min: 2, stackDelta: -1 },
|
||||||
|
{ id: 'not', min: 2, stackDelta: -1 },
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
{ id: 'abs', min: 1, stackDelta: 0 },
|
||||||
|
{ id: 'add', min: 2, stackDelta: -1 },
|
||||||
|
{ id: 'sub', min: 2, stackDelta: -1 },
|
||||||
|
{ id: 'div', min: 2, stackDelta: -1 },
|
||||||
|
null,
|
||||||
|
{ id: 'neg', min: 1, stackDelta: 0 },
|
||||||
|
{ id: 'eq', min: 2, stackDelta: -1 },
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
{ id: 'drop', min: 1, stackDelta: -1 },
|
||||||
|
null,
|
||||||
|
{ id: 'put', min: 2, stackDelta: -2 },
|
||||||
|
{ id: 'get', min: 1, stackDelta: 0 },
|
||||||
|
{ id: 'ifelse', min: 4, stackDelta: -3 },
|
||||||
|
{ id: 'random', min: 0, stackDelta: 1 },
|
||||||
|
{ id: 'mul', min: 2, stackDelta: -1 },
|
||||||
|
null,
|
||||||
|
{ id: 'sqrt', min: 1, stackDelta: 0 },
|
||||||
|
{ id: 'dup', min: 1, stackDelta: 1 },
|
||||||
|
{ id: 'exch', min: 2, stackDelta: 0 },
|
||||||
|
{ id: 'index', min: 2, stackDelta: 0 },
|
||||||
|
{ id: 'roll', min: 3, stackDelta: -2 },
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
{ id: 'hflex', min: 7, resetStack: true },
|
||||||
|
{ id: 'flex', min: 13, resetStack: true },
|
||||||
|
{ id: 'hflex1', min: 9, resetStack: true },
|
||||||
|
{ id: 'flex1', min: 11, resetStack: true }
|
||||||
|
];
|
||||||
|
|
||||||
function CFFParser(file, properties) {
|
function CFFParser(file, properties) {
|
||||||
this.bytes = file.getBytes();
|
this.bytes = file.getBytes();
|
||||||
this.properties = properties;
|
this.properties = properties;
|
||||||
@ -5472,29 +5547,83 @@ var CFFParser = (function CFFParserClosure() {
|
|||||||
},
|
},
|
||||||
parseCharStrings: function CFFParser_parseCharStrings(charStringOffset) {
|
parseCharStrings: function CFFParser_parseCharStrings(charStringOffset) {
|
||||||
var charStrings = this.parseIndex(charStringOffset).obj;
|
var charStrings = this.parseIndex(charStringOffset).obj;
|
||||||
// The CFF specification state that the 'dotsection' command
|
|
||||||
// (12, 0) is deprecated and treated as a no-op, but all Type2
|
|
||||||
// charstrings processors should support them. Unfortunately
|
|
||||||
// the font sanitizer don't. As a workaround the sequence (12, 0)
|
|
||||||
// is replaced by a useless (0, hmoveto).
|
|
||||||
var count = charStrings.count;
|
var count = charStrings.count;
|
||||||
for (var i = 0; i < count; i++) {
|
for (var i = 0; i < count; i++) {
|
||||||
var charstring = charStrings.get(i);
|
var charstring = charStrings.get(i);
|
||||||
|
|
||||||
|
var stackSize = 0;
|
||||||
|
var undefStack = true;
|
||||||
|
var hints = 0;
|
||||||
|
var valid = true;
|
||||||
var data = charstring;
|
var data = charstring;
|
||||||
var length = data.length;
|
var length = data.length;
|
||||||
for (var j = 0; j <= length;) {
|
for (var j = 0; j < length;) {
|
||||||
var value = data[j++];
|
var value = data[j++];
|
||||||
if (value == 12 && data[j++] == 0) {
|
var validationCommand = null;
|
||||||
|
if (value == 12) {
|
||||||
|
var q = data[j++];
|
||||||
|
if (q == 0) {
|
||||||
|
// The CFF specification state that the 'dotsection' command
|
||||||
|
// (12, 0) is deprecated and treated as a no-op, but all Type2
|
||||||
|
// charstrings processors should support them. Unfortunately
|
||||||
|
// the font sanitizer don't. As a workaround the sequence (12, 0)
|
||||||
|
// is replaced by a useless (0, hmoveto).
|
||||||
data[j - 2] = 139;
|
data[j - 2] = 139;
|
||||||
data[j - 1] = 22;
|
data[j - 1] = 22;
|
||||||
} else if (value === 28) {
|
stackSize = 0;
|
||||||
|
} else {
|
||||||
|
validationCommand = CharstringValidationData12[q];
|
||||||
|
}
|
||||||
|
} else if (value === 28) { // number (16 bit)
|
||||||
j += 2;
|
j += 2;
|
||||||
} else if (value >= 247 && value <= 254) {
|
stackSize++;
|
||||||
|
} else if (value == 14) {
|
||||||
|
if (stackSize >= 4) {
|
||||||
|
// TODO fix deprecated endchar construct for Windows
|
||||||
|
stackSize -= 4;
|
||||||
|
}
|
||||||
|
} else if (value >= 32 && value <= 246) { // number
|
||||||
|
stackSize++;
|
||||||
|
} else if (value >= 247 && value <= 254) { // number (+1 bytes)
|
||||||
j++;
|
j++;
|
||||||
} else if (value == 255) {
|
stackSize++;
|
||||||
|
} else if (value == 255) { // number (32 bit)
|
||||||
j += 4;
|
j += 4;
|
||||||
|
stackSize++;
|
||||||
|
} else if (value == 18 || value == 23) {
|
||||||
|
hints += stackSize >> 1;
|
||||||
|
validationCommand = CharstringValidationData[value];
|
||||||
|
} else if (value == 19 || value == 20) {
|
||||||
|
hints += stackSize >> 1;
|
||||||
|
j += (hints + 7) >> 3; // skipping right amount of hints flag data
|
||||||
|
stackSize = 0;
|
||||||
|
} else {
|
||||||
|
validationCommand = CharstringValidationData[value];
|
||||||
}
|
}
|
||||||
|
if (validationCommand) {
|
||||||
|
if ('min' in validationCommand) {
|
||||||
|
if (!undefStack && stackSize < validationCommand.min) {
|
||||||
|
warn('Not enough parameters for ' + validationCommand.id +
|
||||||
|
'; actual: ' + stackSize +
|
||||||
|
', expected: ' + validationCommand.min);
|
||||||
|
valid = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ('stackDelta' in validationCommand) {
|
||||||
|
stackSize += validationCommand.stackDelta;
|
||||||
|
} else if (validationCommand.resetStack) {
|
||||||
|
stackSize = 0;
|
||||||
|
undefStack = false;
|
||||||
|
} else if (validationCommand.undefStack) {
|
||||||
|
stackSize = 0;
|
||||||
|
undefStack = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!valid) {
|
||||||
|
// resetting invalid charstring to single 'endchar'
|
||||||
|
charStrings.set(i, new Uint8Array([14]));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return charStrings;
|
return charStrings;
|
||||||
@ -5760,6 +5889,10 @@ var CFFIndex = (function CFFIndexClosure() {
|
|||||||
this.length += data.length;
|
this.length += data.length;
|
||||||
this.objects.push(data);
|
this.objects.push(data);
|
||||||
},
|
},
|
||||||
|
set: function CFFIndex_set(index, data) {
|
||||||
|
this.length += data.length - this.objects[index].length;
|
||||||
|
this.objects[index] = data;
|
||||||
|
},
|
||||||
get: function CFFIndex_get(index) {
|
get: function CFFIndex_get(index) {
|
||||||
return this.objects[index];
|
return this.objects[index];
|
||||||
},
|
},
|
||||||
|
Loading…
Reference in New Issue
Block a user