summaryrefslogtreecommitdiffstats
path: root/courgette/ensemble.cc
blob: a9b7f38fc9faee6ee2822fb7e45e7a95a1ad2263 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
// Copyright (c) 2009 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.

// *** File comments

#include "courgette/ensemble.h"

#include "base/basictypes.h"
#include "base/logging.h"
#include "base/string_util.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() + "("
      + IntToString(kind()) + ","
      + Uint64ToString(offset_in_ensemble()) + ","
      + 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