summaryrefslogtreecommitdiffstats
path: root/lld/lib/ReaderWriter/PECOFF/OrderPass.h
blob: 9ffe179487ab75ae102c0451f7b85413430ffc40 (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
//===- lib/ReaderWriter/PECOFF/OrderPass.h -------------------------------===//
//
//                             The LLVM Linker
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
///
/// \file \brief This pass sorts atoms by section name, so that they will appear
/// in the correct order in the output.
///
/// In COFF, sections will be merged into one section by the linker if their
/// names are the same after discarding the "$" character and all characters
/// follow it from their names. The characters following the "$" character
/// determines the merge order. Assume there's an object file containing four
/// data sections in the following order.
///
///   - .data$2
///   - .data$3
///   - .data$1
///   - .data
///
/// In this case, the resulting binary should have ".data" section with the
/// contents of ".data", ".data$1", ".data$2" and ".data$3" in that order.
///
//===----------------------------------------------------------------------===//

#ifndef LLD_READER_WRITER_PE_COFF_ORDER_PASS_H
#define LLD_READER_WRITER_PE_COFF_ORDER_PASS_H

#include "Atoms.h"
#include "lld/Core/Parallel.h"
#include "lld/Core/Pass.h"
#include <algorithm>

namespace lld {
namespace pecoff {

static bool compare(const DefinedAtom *lhs, const DefinedAtom *rhs) {
  bool lhsCustom = (lhs->sectionChoice() == DefinedAtom::sectionCustomRequired);
  bool rhsCustom = (rhs->sectionChoice() == DefinedAtom::sectionCustomRequired);
  if (lhsCustom && rhsCustom) {
    int cmp = lhs->customSectionName().compare(rhs->customSectionName());
    if (cmp != 0)
      return cmp < 0;
    return DefinedAtom::compareByPosition(lhs, rhs);
  }
  if (lhsCustom && !rhsCustom)
    return true;
  if (!lhsCustom && rhsCustom)
    return false;
  return DefinedAtom::compareByPosition(lhs, rhs);
}

class OrderPass : public lld::Pass {
public:
  void perform(std::unique_ptr<SimpleFile> &file) override {
    SimpleFile::DefinedAtomRange defined = file->definedAtoms();
    parallel_sort(defined.begin(), defined.end(), compare);
  }
};

} // namespace pecoff
} // namespace lld

#endif
OpenPOWER on IntegriCloud