#include "pdf/font_table_linux.h"
#include <sys/stat.h>
#include <unistd.h>
#include <algorithm>
#include <limits>
#include <memory>
#include "base/numerics/safe_conversions.h"
#include "base/posix/eintr_wrapper.h"
#include "base/sys_byteorder.h"
namespace pdf {
bool GetFontTable(int fd,
uint32_t table_tag,
off_t offset,
uint8_t* output,
size_t* output_length) {
if (offset < 0)
return false;
size_t data_length = 0;
off_t data_offset = 0;
if (table_tag == 0) {
struct stat st;
if (fstat(fd, &st) < 0)
return false;
data_length = base::checked_cast<size_t>(st.st_size);
} else {
uint16_t num_tables;
ssize_t n = HANDLE_EINTR(
pread(fd, &num_tables, sizeof(num_tables), 4 ));
if (n != sizeof(num_tables))
return false;
num_tables = base::NetToHost16(num_tables);
static const size_t kTableEntrySize = 16;
const size_t directory_size = num_tables * kTableEntrySize;
std::unique_ptr<uint8_t[]> table_entries(new uint8_t[directory_size]);
n = HANDLE_EINTR(pread(fd, table_entries.get(), directory_size,
12 ));
if (n != base::checked_cast<ssize_t>(directory_size))
return false;
for (uint16_t i = 0; i < num_tables; ++i) {
uint8_t* entry = table_entries.get() + i * kTableEntrySize;
uint32_t tag = *reinterpret_cast<uint32_t*>(entry);
if (tag == table_tag) {
data_offset =
base::NetToHost32(*reinterpret_cast<uint32_t*>(entry + 8));
data_length =
base::NetToHost32(*reinterpret_cast<uint32_t*>(entry + 12));
break;
}
}
}
if (!data_length)
return false;
offset = std::min(offset, base::checked_cast<off_t>(data_length));
static const off_t kMaxPositiveOffset32 = 0x7FFFFFFF;
if ((offset > kMaxPositiveOffset32 / 2) ||
(data_offset > kMaxPositiveOffset32 / 2))
return false;
data_offset += offset;
data_length -= offset;
if (output) {
data_length = std::min(data_length, *output_length);
ssize_t n = HANDLE_EINTR(pread(fd, output, data_length, data_offset));
if (n != base::checked_cast<ssize_t>(data_length))
return false;
}
*output_length = data_length;
return true;
}
}