| // | 
 | // Copyright (C) 2009 The Android Open Source Project | 
 | // | 
 | // Licensed under the Apache License, Version 2.0 (the "License"); | 
 | // you may not use this file except in compliance with the License. | 
 | // You may obtain a copy of the License at | 
 | // | 
 | //      http://www.apache.org/licenses/LICENSE-2.0 | 
 | // | 
 | // Unless required by applicable law or agreed to in writing, software | 
 | // distributed under the License is distributed on an "AS IS" BASIS, | 
 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | // See the License for the specific language governing permissions and | 
 | // limitations under the License. | 
 | // | 
 |  | 
 | #include "update_engine/payload_consumer/bzip_extent_writer.h" | 
 |  | 
 | using google::protobuf::RepeatedPtrField; | 
 |  | 
 | namespace chromeos_update_engine { | 
 |  | 
 | namespace { | 
 | const brillo::Blob::size_type kOutputBufferLength = 16 * 1024; | 
 | } | 
 |  | 
 | bool BzipExtentWriter::Init(FileDescriptorPtr fd, | 
 |                             const RepeatedPtrField<Extent>& extents, | 
 |                             uint32_t block_size) { | 
 |   // Init bzip2 stream | 
 |   int rc = BZ2_bzDecompressInit(&stream_, | 
 |                                 0,   // verbosity. (0 == silent) | 
 |                                 0);  // 0 = faster algo, more memory | 
 |  | 
 |   TEST_AND_RETURN_FALSE(rc == BZ_OK); | 
 |  | 
 |   return next_->Init(fd, extents, block_size); | 
 | } | 
 |  | 
 | bool BzipExtentWriter::Write(const void* bytes, size_t count) { | 
 |   brillo::Blob output_buffer(kOutputBufferLength); | 
 |  | 
 |   // Copy the input data into |input_buffer_| only if |input_buffer_| already | 
 |   // contains unconsumed data. Otherwise, process the data directly from the | 
 |   // source. | 
 |   const uint8_t* input = reinterpret_cast<const uint8_t*>(bytes); | 
 |   const uint8_t* input_end = input + count; | 
 |   if (!input_buffer_.empty()) { | 
 |     input_buffer_.insert(input_buffer_.end(), input, input_end); | 
 |     input = input_buffer_.data(); | 
 |     input_end = input + input_buffer_.size(); | 
 |   } | 
 |   stream_.next_in = reinterpret_cast<char*>(const_cast<uint8_t*>(input)); | 
 |   stream_.avail_in = input_end - input; | 
 |  | 
 |   for (;;) { | 
 |     stream_.next_out = reinterpret_cast<char*>(output_buffer.data()); | 
 |     stream_.avail_out = output_buffer.size(); | 
 |  | 
 |     int rc = BZ2_bzDecompress(&stream_); | 
 |     TEST_AND_RETURN_FALSE(rc == BZ_OK || rc == BZ_STREAM_END); | 
 |  | 
 |     if (stream_.avail_out == output_buffer.size()) | 
 |       break;  // got no new bytes | 
 |  | 
 |     TEST_AND_RETURN_FALSE( | 
 |         next_->Write(output_buffer.data(), | 
 |                      output_buffer.size() - stream_.avail_out)); | 
 |  | 
 |     if (rc == BZ_STREAM_END) | 
 |       CHECK_EQ(stream_.avail_in, 0u); | 
 |     if (stream_.avail_in == 0) | 
 |       break;  // no more input to process | 
 |   } | 
 |  | 
 |   // Store unconsumed data (if any) in |input_buffer_|. | 
 |   if (stream_.avail_in || !input_buffer_.empty()) { | 
 |     brillo::Blob new_input_buffer(input_end - stream_.avail_in, input_end); | 
 |     new_input_buffer.swap(input_buffer_); | 
 |   } | 
 |  | 
 |   return true; | 
 | } | 
 |  | 
 | bool BzipExtentWriter::EndImpl() { | 
 |   TEST_AND_RETURN_FALSE(input_buffer_.empty()); | 
 |   TEST_AND_RETURN_FALSE(BZ2_bzDecompressEnd(&stream_) == BZ_OK); | 
 |   return next_->End(); | 
 | } | 
 |  | 
 | }  // namespace chromeos_update_engine |