aboutsummaryrefslogtreecommitdiffstats
path: root/contrib/libs/llvm12/tools/llvm-objcopy/wasm/WasmObjcopy.cpp
blob: fd83334ed927a293e0da0a2690846d1c8ca563f3 (plain) (blame)
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
106
107
108
109
110
111
112
113
//===- WasmObjcopy.cpp ----------------------------------------------------===// 
// 
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 
// See https://llvm.org/LICENSE.txt for license information. 
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 
// 
//===----------------------------------------------------------------------===// 
 
#include "WasmObjcopy.h" 
#include "Buffer.h" 
#include "CopyConfig.h" 
#include "Object.h" 
#include "Reader.h" 
#include "Writer.h" 
#include "llvm/Support/Errc.h" 
 
namespace llvm { 
namespace objcopy { 
namespace wasm { 
 
using namespace object; 
 
static Error dumpSectionToFile(StringRef SecName, StringRef Filename, 
                               Object &Obj) { 
  for (const Section &Sec : Obj.Sections) { 
    if (Sec.Name == SecName) { 
      ArrayRef<uint8_t> Contents = Sec.Contents; 
      Expected<std::unique_ptr<FileOutputBuffer>> BufferOrErr = 
          FileOutputBuffer::create(Filename, Contents.size()); 
      if (!BufferOrErr) 
        return BufferOrErr.takeError(); 
      std::unique_ptr<FileOutputBuffer> Buf = std::move(*BufferOrErr); 
      std::copy(Contents.begin(), Contents.end(), Buf->getBufferStart()); 
      if (Error E = Buf->commit()) 
        return E; 
      return Error::success(); 
    } 
  } 
  return createStringError(errc::invalid_argument, "section '%s' not found", 
                           SecName.str().c_str()); 
} 
static Error handleArgs(const CopyConfig &Config, Object &Obj) { 
  // Only support AddSection, DumpSection, RemoveSection for now. 
  for (StringRef Flag : Config.DumpSection) { 
    StringRef SecName; 
    StringRef FileName; 
    std::tie(SecName, FileName) = Flag.split("="); 
    if (Error E = dumpSectionToFile(SecName, FileName, Obj)) 
      return createFileError(FileName, std::move(E)); 
  } 
 
  Obj.removeSections([&Config](const Section &Sec) { 
    if (Config.ToRemove.matches(Sec.Name)) 
      return true; 
    return false; 
  }); 
 
  for (StringRef Flag : Config.AddSection) { 
    StringRef SecName, FileName; 
    std::tie(SecName, FileName) = Flag.split("="); 
    ErrorOr<std::unique_ptr<MemoryBuffer>> BufOrErr = 
        MemoryBuffer::getFile(FileName); 
    if (!BufOrErr) 
      return createFileError(FileName, errorCodeToError(BufOrErr.getError())); 
    Section Sec; 
    Sec.SectionType = llvm::wasm::WASM_SEC_CUSTOM; 
    Sec.Name = SecName; 
    std::unique_ptr<MemoryBuffer> Buf = std::move(*BufOrErr); 
    Sec.Contents = makeArrayRef<uint8_t>( 
        reinterpret_cast<const uint8_t *>(Buf->getBufferStart()), 
        Buf->getBufferSize()); 
    Obj.addSectionWithOwnedContents(Sec, std::move(Buf)); 
  } 
 
  if (!Config.AddGnuDebugLink.empty() || !Config.BuildIdLinkDir.empty() || 
      Config.BuildIdLinkInput || Config.BuildIdLinkOutput || 
      Config.ExtractPartition || !Config.SplitDWO.empty() || 
      !Config.SymbolsPrefix.empty() || !Config.AllocSectionsPrefix.empty() || 
      Config.DiscardMode != DiscardType::None || Config.NewSymbolVisibility || 
      !Config.SymbolsToAdd.empty() || !Config.RPathToAdd.empty() || 
      !Config.OnlySection.empty() || !Config.SymbolsToGlobalize.empty() || 
      !Config.SymbolsToKeep.empty() || !Config.SymbolsToLocalize.empty() || 
      !Config.SymbolsToRemove.empty() || 
      !Config.UnneededSymbolsToRemove.empty() || 
      !Config.SymbolsToWeaken.empty() || !Config.SymbolsToKeepGlobal.empty() || 
      !Config.SectionsToRename.empty() || !Config.SetSectionAlignment.empty() || 
      !Config.SetSectionFlags.empty() || !Config.SymbolsToRename.empty()) { 
    return createStringError( 
        llvm::errc::invalid_argument, 
        "only add-section, dump-section, and remove-section are supported"); 
  } 
  return Error::success(); 
} 
 
Error executeObjcopyOnBinary(const CopyConfig &Config, 
                             object::WasmObjectFile &In, Buffer &Out) { 
  Reader TheReader(In); 
  Expected<std::unique_ptr<Object>> ObjOrErr = TheReader.create(); 
  if (!ObjOrErr) 
    return createFileError(Config.InputFilename, ObjOrErr.takeError()); 
  Object *Obj = ObjOrErr->get(); 
  assert(Obj && "Unable to deserialize Wasm object"); 
  if (Error E = handleArgs(Config, *Obj)) 
    return E; 
  Writer TheWriter(*Obj, Out); 
  if (Error E = TheWriter.write()) 
    return createFileError(Config.OutputFilename, std::move(E)); 
  return Error::success(); 
} 
 
} // end namespace wasm 
} // end namespace objcopy 
} // end namespace llvm