1 // Copyright 2017 PDFium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "core/fpdfapi/parser/cpdf_read_validator.h"
6
7 #include <algorithm>
8
9 #include "core/fpdfapi/parser/cpdf_stream.h"
10 #include "core/fxcrt/fx_safe_types.h"
11 #include "third_party/base/logging.h"
12
13 namespace {
14
15 constexpr FX_FILESIZE kAlignBlockValue = CPDF_Stream::kFileBufSize;
16
AlignDown(FX_FILESIZE offset)17 FX_FILESIZE AlignDown(FX_FILESIZE offset) {
18 return offset > 0 ? (offset - offset % kAlignBlockValue) : 0;
19 }
20
AlignUp(FX_FILESIZE offset)21 FX_FILESIZE AlignUp(FX_FILESIZE offset) {
22 FX_SAFE_FILESIZE safe_result = AlignDown(offset);
23 safe_result += kAlignBlockValue;
24 return safe_result.ValueOrDefault(offset);
25 }
26
27 } // namespace
28
Session(const RetainPtr<CPDF_ReadValidator> & validator)29 CPDF_ReadValidator::Session::Session(
30 const RetainPtr<CPDF_ReadValidator>& validator)
31 : validator_(validator.BackPointer()) {
32 ASSERT(validator_);
33 saved_read_error_ = validator_->read_error_;
34 saved_has_unavailable_data_ = validator_->has_unavailable_data_;
35 validator_->ResetErrors();
36 }
37
~Session()38 CPDF_ReadValidator::Session::~Session() {
39 validator_->read_error_ |= saved_read_error_;
40 validator_->has_unavailable_data_ |= saved_has_unavailable_data_;
41 }
42
CPDF_ReadValidator(const RetainPtr<IFX_SeekableReadStream> & file_read,CPDF_DataAvail::FileAvail * file_avail)43 CPDF_ReadValidator::CPDF_ReadValidator(
44 const RetainPtr<IFX_SeekableReadStream>& file_read,
45 CPDF_DataAvail::FileAvail* file_avail)
46 : file_read_(file_read),
47 file_avail_(file_avail),
48 read_error_(false),
49 has_unavailable_data_(false),
50 whole_file_already_available_(false),
51 file_size_(file_read->GetSize()) {}
52
~CPDF_ReadValidator()53 CPDF_ReadValidator::~CPDF_ReadValidator() {}
54
ResetErrors()55 void CPDF_ReadValidator::ResetErrors() {
56 read_error_ = false;
57 has_unavailable_data_ = false;
58 }
59
ReadBlockAtOffset(void * buffer,FX_FILESIZE offset,size_t size)60 bool CPDF_ReadValidator::ReadBlockAtOffset(void* buffer,
61 FX_FILESIZE offset,
62 size_t size) {
63 FX_SAFE_FILESIZE end_offset = offset;
64 end_offset += size;
65 if (!end_offset.IsValid() || end_offset.ValueOrDie() > file_size_)
66 return false;
67
68 if (!IsDataRangeAvailable(offset, size)) {
69 ScheduleDownload(offset, size);
70 return false;
71 }
72
73 if (file_read_->ReadBlockAtOffset(buffer, offset, size))
74 return true;
75
76 read_error_ = true;
77 ScheduleDownload(offset, size);
78 return false;
79 }
80
GetSize()81 FX_FILESIZE CPDF_ReadValidator::GetSize() {
82 return file_size_;
83 }
84
ScheduleDownload(FX_FILESIZE offset,size_t size)85 void CPDF_ReadValidator::ScheduleDownload(FX_FILESIZE offset, size_t size) {
86 has_unavailable_data_ = true;
87 if (!hints_ || size == 0)
88 return;
89
90 const FX_FILESIZE start_segment_offset = AlignDown(offset);
91 FX_SAFE_FILESIZE end_segment_offset = offset;
92 end_segment_offset += size;
93 if (!end_segment_offset.IsValid()) {
94 NOTREACHED();
95 return;
96 }
97 end_segment_offset =
98 std::min(file_size_, AlignUp(end_segment_offset.ValueOrDie()));
99
100 FX_SAFE_SIZE_T segment_size = end_segment_offset;
101 segment_size -= start_segment_offset;
102 if (!segment_size.IsValid()) {
103 NOTREACHED();
104 return;
105 }
106 hints_->AddSegment(start_segment_offset, segment_size.ValueOrDie());
107 }
108
IsDataRangeAvailable(FX_FILESIZE offset,size_t size) const109 bool CPDF_ReadValidator::IsDataRangeAvailable(FX_FILESIZE offset,
110 size_t size) const {
111 return whole_file_already_available_ || !file_avail_ ||
112 file_avail_->IsDataAvail(offset, size);
113 }
114
IsWholeFileAvailable()115 bool CPDF_ReadValidator::IsWholeFileAvailable() {
116 const FX_SAFE_SIZE_T safe_size = file_size_;
117 whole_file_already_available_ =
118 whole_file_already_available_ ||
119 (safe_size.IsValid() ? IsDataRangeAvailable(0, safe_size.ValueOrDie())
120 : false);
121
122 return whole_file_already_available_;
123 }
124
CheckDataRangeAndRequestIfUnavailable(FX_FILESIZE offset,size_t size)125 bool CPDF_ReadValidator::CheckDataRangeAndRequestIfUnavailable(
126 FX_FILESIZE offset,
127 size_t size) {
128 if (offset > file_size_)
129 return true;
130
131 FX_SAFE_FILESIZE end_segment_offset = offset;
132 end_segment_offset += size;
133 // Increase checked range to allow CPDF_SyntaxParser read whole buffer.
134 end_segment_offset += CPDF_Stream::kFileBufSize;
135 if (!end_segment_offset.IsValid()) {
136 NOTREACHED();
137 return false;
138 }
139 end_segment_offset = std::min(
140 file_size_, static_cast<FX_FILESIZE>(end_segment_offset.ValueOrDie()));
141 FX_SAFE_SIZE_T segment_size = end_segment_offset;
142 segment_size -= offset;
143 if (!segment_size.IsValid()) {
144 NOTREACHED();
145 return false;
146 }
147
148 if (IsDataRangeAvailable(offset, segment_size.ValueOrDie()))
149 return true;
150
151 ScheduleDownload(offset, segment_size.ValueOrDie());
152 return false;
153 }
154
CheckWholeFileAndRequestIfUnavailable()155 bool CPDF_ReadValidator::CheckWholeFileAndRequestIfUnavailable() {
156 if (IsWholeFileAvailable())
157 return true;
158
159 const FX_SAFE_SIZE_T safe_size = file_size_;
160 if (safe_size.IsValid())
161 ScheduleDownload(0, safe_size.ValueOrDie());
162
163 return false;
164 }
165