blob: 69e07a7a52567970bc4da53466d18a0e160da718 [file] [log] [blame]
// Copyright (c) 2010 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "courgette/ensemble.h"
#include "base/basictypes.h"
#include "base/string_number_conversions.h"
#include "courgette/image_info.h"
#include "courgette/region.h"
#include "courgette/streams.h"
#include "courgette/simple_delta.h"
namespace courgette {
Element::Element(Kind kind, Ensemble* ensemble, const Region& region)
: kind_(kind), ensemble_(ensemble), region_(region) {
}
std::string Element::Name() const {
return ensemble_->name() + "("
+ base::IntToString(kind()) + ","
+ base::Uint64ToString(offset_in_ensemble()) + ","
+ base::Uint64ToString(region().length()) + ")";
}
// A subclass of Element that has a PEInfo.
class ElementWinPE : public Element {
public:
ElementWinPE(Kind kind, Ensemble* ensemble, const Region& region,
PEInfo* info)
: Element(kind, ensemble, region),
pe_info_(info) {
}
virtual PEInfo* GetPEInfo() const { return pe_info_; }
protected:
~ElementWinPE() { delete pe_info_; }
private:
PEInfo* pe_info_; // Owned by |this|.
};
// Scans the Ensemble's region, sniffing out Elements. We assume that the
// elements do not overlap.
Status Ensemble::FindEmbeddedElements() {
size_t length = region_.length();
const uint8* start = region_.start();
size_t position = 0;
while (position < length) {
// Quick test; Windows executables begin with 'MZ'.
if (start[position] == 'M' &&
position + 1 < length && start[position + 1] == 'Z') {
courgette::PEInfo *info = new courgette::PEInfo();
info->Init(start + position, length - position);
if (info->ParseHeader()) {
Region region(start + position, info->length());
if (info->has_text_section()) {
if (info->is_32bit()) {
Element* element = new ElementWinPE(Element::WIN32_X86_WITH_CODE,
this, region, info);
owned_elements_.push_back(element);
elements_.push_back(element);
position += region.length();
continue;
}
// TODO(sra): Extend to 64-bit executables.
}
// If we had a clever transformation for resource-only executables we
// should identify the suitable elements here:
if (!info->has_text_section() && false) {
Element* element = new ElementWinPE(Element::WIN32_NOCODE,
this, region, info);
owned_elements_.push_back(element);
elements_.push_back(element);
position += region.length();
continue;
}
}
delete info;
}
// This is where to add new formats, e.g. Linux executables, Dalvik
// executables etc.
// No Element found at current position.
++position;
}
return C_OK;
}
Ensemble::~Ensemble() {
for (size_t i = 0; i < owned_elements_.size(); ++i)
delete owned_elements_[i];
}
} // namespace