mirror of
https://github.com/qpdf/qpdf.git
synced 2025-01-26 00:28:26 +00:00
414 lines
13 KiB
C++
414 lines
13 KiB
C++
#include <qpdf/QIntC.hh>
|
|
#include <qpdf/QPDF.hh>
|
|
#include <qpdf/QPDFXRefEntry.hh>
|
|
#include <qpdf/QUtil.hh>
|
|
#include <cstdio>
|
|
#include <cstring>
|
|
#include <iostream>
|
|
#include <regex>
|
|
#include <string_view>
|
|
|
|
static char const* whoami = nullptr;
|
|
|
|
static void
|
|
usage()
|
|
{
|
|
std::cerr << "Usage: " << whoami << " [filename]" << std::endl;
|
|
exit(2);
|
|
}
|
|
|
|
class QdfFixer
|
|
{
|
|
public:
|
|
QdfFixer(std::string const& filename);
|
|
void processLines(std::string const& input);
|
|
|
|
private:
|
|
void fatal(std::string const&);
|
|
void checkObjId(std::string const& obj_id);
|
|
void adjustOstreamXref();
|
|
void writeOstream();
|
|
void writeBinary(unsigned long long val, size_t bytes);
|
|
|
|
std::string filename;
|
|
enum {
|
|
st_top,
|
|
st_in_obj,
|
|
st_in_stream,
|
|
st_after_stream,
|
|
st_in_ostream_dict,
|
|
st_in_ostream_offsets,
|
|
st_in_ostream_outer,
|
|
st_in_ostream_obj,
|
|
st_in_xref_stream_dict,
|
|
st_in_length,
|
|
st_at_xref,
|
|
st_before_trailer,
|
|
st_in_trailer,
|
|
st_done,
|
|
} state{st_top};
|
|
|
|
size_t lineno{0};
|
|
qpdf_offset_t offset{0};
|
|
qpdf_offset_t last_offset{0};
|
|
int last_obj{0};
|
|
std::vector<QPDFXRefEntry> xref;
|
|
qpdf_offset_t stream_start{0};
|
|
size_t stream_length{0};
|
|
qpdf_offset_t xref_offset{0};
|
|
size_t xref_f1_nbytes{0};
|
|
size_t xref_f2_nbytes{0};
|
|
size_t xref_size{0};
|
|
std::vector<std::string_view> ostream;
|
|
std::vector<qpdf_offset_t> ostream_offsets;
|
|
std::vector<std::string_view> ostream_discarded;
|
|
size_t ostream_idx{0};
|
|
int ostream_id{0};
|
|
std::string ostream_extends;
|
|
};
|
|
|
|
QdfFixer::QdfFixer(std::string const& filename) :
|
|
filename(filename)
|
|
{
|
|
}
|
|
|
|
void
|
|
QdfFixer::fatal(std::string const& msg)
|
|
{
|
|
std::cerr << msg << std::endl;
|
|
exit(2);
|
|
}
|
|
|
|
void
|
|
QdfFixer::processLines(std::string const& input)
|
|
{
|
|
using namespace std::literals;
|
|
|
|
static const std::regex re_n_0_obj("^(\\d+) 0 obj\n$");
|
|
static const std::regex re_extends("/Extends (\\d+ 0 R)");
|
|
static const std::regex re_ostream_obj("^%% Object stream: object (\\d+)");
|
|
static const std::regex re_num("^\\d+\n$");
|
|
static const std::regex re_size_n("^ /Size \\d+\n$");
|
|
|
|
auto sv_diff = [](size_t i) { return static_cast<std::string_view::difference_type>(i); };
|
|
|
|
lineno = 0;
|
|
bool more = true;
|
|
auto len_line = sv_diff(0);
|
|
|
|
std::string_view line;
|
|
std::string_view input_view{input.data(), input.size()};
|
|
size_t offs = 0;
|
|
|
|
auto b_line = input.cbegin();
|
|
std::smatch m;
|
|
auto const matches = [&m, &b_line, &len_line](std::regex const& r) {
|
|
return std::regex_search(b_line, b_line + len_line, m, r);
|
|
};
|
|
|
|
while (more) {
|
|
++lineno;
|
|
last_offset = offset;
|
|
b_line += len_line;
|
|
|
|
offs = input_view.find('\n');
|
|
if (offs == std::string::npos) {
|
|
more = false;
|
|
line = input_view;
|
|
} else {
|
|
offs++;
|
|
line = input_view.substr(0, offs);
|
|
input_view.remove_prefix(offs);
|
|
}
|
|
len_line = sv_diff(line.size());
|
|
offset += len_line;
|
|
|
|
if (state == st_top) {
|
|
if (matches(re_n_0_obj)) {
|
|
checkObjId(m[1].str());
|
|
state = st_in_obj;
|
|
} else if (line.compare("xref\n"sv) == 0) {
|
|
xref_offset = last_offset;
|
|
state = st_at_xref;
|
|
}
|
|
std::cout << line;
|
|
} else if (state == st_in_obj) {
|
|
std::cout << line;
|
|
if (line.compare("stream\n"sv) == 0) {
|
|
state = st_in_stream;
|
|
stream_start = offset;
|
|
} else if (line.compare("endobj\n"sv) == 0) {
|
|
state = st_top;
|
|
} else if (line.find("/Type /ObjStm"sv) != line.npos) {
|
|
state = st_in_ostream_dict;
|
|
ostream_id = last_obj;
|
|
} else if (line.find("/Type /XRef"sv) != line.npos) {
|
|
xref_offset = xref.back().getOffset();
|
|
xref_f1_nbytes = 0;
|
|
auto t = xref_offset;
|
|
while (t) {
|
|
t >>= 8;
|
|
++xref_f1_nbytes;
|
|
}
|
|
// Figure out how many bytes we need for ostream
|
|
// index. Make sure we get at least 1 byte even if
|
|
// there are no object streams.
|
|
int max_objects = 1;
|
|
for (auto const& e: xref) {
|
|
if ((e.getType() == 2) && (e.getObjStreamIndex() > max_objects)) {
|
|
max_objects = e.getObjStreamIndex();
|
|
}
|
|
}
|
|
while (max_objects) {
|
|
max_objects >>= 8;
|
|
++xref_f2_nbytes;
|
|
}
|
|
auto esize = 1 + xref_f1_nbytes + xref_f2_nbytes;
|
|
xref_size = 1 + xref.size();
|
|
auto length = xref_size * esize;
|
|
std::cout << " /Length " << length << "\n"
|
|
<< " /W [ 1 " << xref_f1_nbytes << " " << xref_f2_nbytes << " ]\n";
|
|
state = st_in_xref_stream_dict;
|
|
}
|
|
} else if (state == st_in_ostream_dict) {
|
|
if (line.compare("stream\n"sv) == 0) {
|
|
state = st_in_ostream_offsets;
|
|
} else {
|
|
ostream_discarded.push_back(line);
|
|
if (matches(re_extends)) {
|
|
ostream_extends = m[1].str();
|
|
}
|
|
}
|
|
// discard line
|
|
} else if (state == st_in_ostream_offsets) {
|
|
if (matches(re_ostream_obj)) {
|
|
checkObjId(m[1].str());
|
|
stream_start = last_offset;
|
|
state = st_in_ostream_outer;
|
|
ostream.push_back(line);
|
|
} else {
|
|
ostream_discarded.push_back(line);
|
|
}
|
|
// discard line
|
|
} else if (state == st_in_ostream_outer) {
|
|
adjustOstreamXref();
|
|
ostream_offsets.push_back(last_offset - stream_start);
|
|
state = st_in_ostream_obj;
|
|
ostream.push_back(line);
|
|
} else if (state == st_in_ostream_obj) {
|
|
ostream.push_back(line);
|
|
if (matches(re_ostream_obj)) {
|
|
checkObjId(m[1].str());
|
|
state = st_in_ostream_outer;
|
|
} else if (line.compare("endstream\n"sv) == 0) {
|
|
stream_length = QIntC::to_size(last_offset - stream_start);
|
|
writeOstream();
|
|
state = st_in_obj;
|
|
}
|
|
} else if (state == st_in_xref_stream_dict) {
|
|
if ((line.find("/Length"sv) != line.npos) || (line.find("/W"sv) != line.npos)) {
|
|
// already printed
|
|
} else if (line.find("/Size"sv) != line.npos) {
|
|
auto xref_size = 1 + xref.size();
|
|
std::cout << " /Size " << xref_size << "\n";
|
|
} else {
|
|
std::cout << line;
|
|
}
|
|
if (line.compare("stream\n"sv) == 0) {
|
|
writeBinary(0, 1);
|
|
writeBinary(0, xref_f1_nbytes);
|
|
writeBinary(0, xref_f2_nbytes);
|
|
for (auto const& x: xref) {
|
|
unsigned long long f1 = 0;
|
|
unsigned long long f2 = 0;
|
|
unsigned int type = QIntC::to_uint(x.getType());
|
|
if (1 == type) {
|
|
f1 = QIntC::to_ulonglong(x.getOffset());
|
|
} else {
|
|
f1 = QIntC::to_ulonglong(x.getObjStreamNumber());
|
|
f2 = QIntC::to_ulonglong(x.getObjStreamIndex());
|
|
}
|
|
writeBinary(type, 1);
|
|
writeBinary(f1, xref_f1_nbytes);
|
|
writeBinary(f2, xref_f2_nbytes);
|
|
}
|
|
std::cout << "\nendstream\nendobj\n\n"
|
|
<< "startxref\n"
|
|
<< xref_offset << "\n%%EOF\n";
|
|
state = st_done;
|
|
}
|
|
} else if (state == st_in_stream) {
|
|
if (line.compare("endstream\n"sv) == 0) {
|
|
stream_length = QIntC::to_size(last_offset - stream_start);
|
|
state = st_after_stream;
|
|
}
|
|
std::cout << line;
|
|
} else if (state == st_after_stream) {
|
|
if (line.compare("%QDF: ignore_newline\n"sv) == 0) {
|
|
if (stream_length > 0) {
|
|
--stream_length;
|
|
}
|
|
} else if (matches(re_n_0_obj)) {
|
|
checkObjId(m[1].str());
|
|
state = st_in_length;
|
|
}
|
|
std::cout << line;
|
|
} else if (state == st_in_length) {
|
|
if (!matches(re_num)) {
|
|
fatal(filename + ":" + std::to_string(lineno) + ": expected integer");
|
|
}
|
|
std::string new_length = std::to_string(stream_length) + "\n";
|
|
offset -= QIntC::to_offset(line.length());
|
|
offset += QIntC::to_offset(new_length.length());
|
|
std::cout << new_length;
|
|
state = st_top;
|
|
} else if (state == st_at_xref) {
|
|
auto n = xref.size();
|
|
std::cout << "0 " << 1 + n << "\n0000000000 65535 f \n";
|
|
for (auto const& e: xref) {
|
|
std::cout << QUtil::int_to_string(e.getOffset(), 10) << " 00000 n \n";
|
|
}
|
|
state = st_before_trailer;
|
|
} else if (state == st_before_trailer) {
|
|
if (line.compare("trailer <<\n"sv) == 0) {
|
|
std::cout << line;
|
|
state = st_in_trailer;
|
|
}
|
|
// no output
|
|
} else if (state == st_in_trailer) {
|
|
if (matches(re_size_n)) {
|
|
std::cout << " /Size " << 1 + xref.size() << "\n";
|
|
} else {
|
|
std::cout << line;
|
|
}
|
|
if (line.compare(">>\n"sv) == 0) {
|
|
std::cout << "startxref\n" << xref_offset << "\n%%EOF\n";
|
|
state = st_done;
|
|
}
|
|
} else if (state == st_done) {
|
|
// ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
QdfFixer::checkObjId(std::string const& cur_obj_str)
|
|
{
|
|
if (std::stoi(cur_obj_str) != ++last_obj) {
|
|
fatal(
|
|
filename + ":" + std::to_string(lineno) + ": expected object " +
|
|
std::to_string(last_obj));
|
|
}
|
|
xref.push_back(QPDFXRefEntry(1, last_offset, 0));
|
|
}
|
|
|
|
void
|
|
QdfFixer::adjustOstreamXref()
|
|
{
|
|
xref.back() = QPDFXRefEntry(2, ostream_id, QIntC::to_int(ostream_idx++));
|
|
}
|
|
|
|
void
|
|
QdfFixer::writeOstream()
|
|
{
|
|
auto first = ostream_offsets.at(0);
|
|
auto onum = ostream_id;
|
|
std::string offsets;
|
|
auto n = ostream_offsets.size();
|
|
for (auto iter: ostream_offsets) {
|
|
iter -= QIntC::to_offset(first);
|
|
++onum;
|
|
offsets += std::to_string(onum) + " " + std::to_string(iter) + "\n";
|
|
}
|
|
auto offset_adjust = QIntC::to_offset(offsets.size());
|
|
first += offset_adjust;
|
|
stream_length += QIntC::to_size(offset_adjust);
|
|
std::string dict_data = "";
|
|
dict_data += " /Length " + std::to_string(stream_length) + "\n";
|
|
dict_data += " /N " + std::to_string(n) + "\n";
|
|
dict_data += " /First " + std::to_string(first) + "\n";
|
|
if (!ostream_extends.empty()) {
|
|
dict_data += " /Extends " + ostream_extends + "\n";
|
|
}
|
|
dict_data += ">>\n";
|
|
offset_adjust += QIntC::to_offset(dict_data.length());
|
|
std::cout << dict_data << "stream\n" << offsets;
|
|
for (auto const& o: ostream) {
|
|
std::cout << o;
|
|
}
|
|
|
|
for (auto const& o: ostream_discarded) {
|
|
offset -= QIntC::to_offset(o.length());
|
|
}
|
|
offset += offset_adjust;
|
|
|
|
ostream_idx = 0;
|
|
ostream_id = 0;
|
|
ostream.clear();
|
|
ostream_offsets.clear();
|
|
ostream_discarded.clear();
|
|
ostream_extends.clear();
|
|
}
|
|
|
|
void
|
|
QdfFixer::writeBinary(unsigned long long val, size_t bytes)
|
|
{
|
|
if (bytes > sizeof(unsigned long long)) {
|
|
throw std::logic_error("fix-qdf::writeBinary called with too many bytes");
|
|
}
|
|
std::string data(bytes, '\0');
|
|
for (auto i = bytes; i > 0; --i) {
|
|
data[i - 1] = static_cast<char>(val & 0xff); // i.e. val % 256
|
|
val >>= 8; // i.e. val = val / 256
|
|
}
|
|
std::cout << data;
|
|
}
|
|
|
|
static int
|
|
realmain(int argc, char* argv[])
|
|
{
|
|
whoami = QUtil::getWhoami(argv[0]);
|
|
QUtil::setLineBuf(stdout);
|
|
char const* filename = nullptr;
|
|
if (argc > 2) {
|
|
usage();
|
|
} else if ((argc > 1) && (strcmp(argv[1], "--version") == 0)) {
|
|
std::cout << whoami << " from qpdf version " << QPDF::QPDFVersion() << std::endl;
|
|
return 0;
|
|
} else if ((argc > 1) && (strcmp(argv[1], "--help") == 0)) {
|
|
usage();
|
|
} else if (argc == 2) {
|
|
filename = argv[1];
|
|
}
|
|
std::string input;
|
|
if (filename == nullptr) {
|
|
filename = "standard input";
|
|
QUtil::binary_stdin();
|
|
input = QUtil::read_file_into_string(stdin);
|
|
} else {
|
|
input = QUtil::read_file_into_string(filename);
|
|
}
|
|
QUtil::binary_stdout();
|
|
QdfFixer qf(filename);
|
|
qf.processLines(input);
|
|
return 0;
|
|
}
|
|
|
|
#ifdef WINDOWS_WMAIN
|
|
|
|
extern "C" int
|
|
wmain(int argc, wchar_t* argv[])
|
|
{
|
|
return QUtil::call_main_from_wmain(argc, argv, realmain);
|
|
}
|
|
|
|
#else
|
|
|
|
int
|
|
main(int argc, char* argv[])
|
|
{
|
|
return realmain(argc, argv);
|
|
}
|
|
|
|
#endif
|