1 /*
2  * Copyright 2013, The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *     http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include <cassert>
18 #include <cstdlib>
19 #include <fcntl.h>
20 #include <fstream>
21 #include <sstream>
22 #include <stdint.h>
23 #include <unistd.h>
24 #include "Abcc.h"
25 
26 #if !defined(_WIN32)
27 #include <sys/mman.h>
28 #else
29 #include "mman.h"
30 #endif
31 
32 #include <sys/types.h>
33 #include <sys/stat.h>
34 using namespace abcc;
35 
TargetAbi(const std::string & abi)36 TargetAbi::TargetAbi(const std::string &abi) {
37   if (abi == "armeabi-v7a" || abi == "armeabi-v7a-hard") //ToDo: support armeabi-v7a-hard
38     mAbi = ARMEABI_V7A;
39   else if (abi == "armeabi")
40     mAbi = ARMEABI;
41   else if (abi == "x86")
42     mAbi = X86;
43   else if (abi == "mips")
44     mAbi = MIPS;
45   else if (abi == "arm64-v8a")
46     mAbi = ARM64_V8A;
47   else if (abi == "x86_64")
48     mAbi = X86_64;
49   else if (abi == "mips64")
50     mAbi = MIPS64;
51   else {
52     assert (false && "Unknown abi for abcc. Check your --abi flag.");
53     exit (1);
54   }
55 }
56 
57 
BitcodeInfo(const std::string & bc)58 BitcodeInfo::BitcodeInfo(const std::string &bc)
59   : mShared(false), mStatic(false), mBCPath(bc) {
60   std::string stem = mBCPath.substr(0, mBCPath.rfind("."));
61   mTargetBCPath = stem + "-target.bc";
62   mObjPath = stem + ".o";
63   mOutPath = stem;  // If shared, we will add .so after readWrapper
64   mSOName = mBCPath.substr(mBCPath.rfind("/") + 1);
65 }
66 
readWrapper(BitcodeCompiler & compiler)67 int BitcodeInfo::readWrapper(BitcodeCompiler &compiler) {
68   int fd = open(mBCPath.c_str(), O_RDONLY);
69 
70   if (fd < 0) {
71     return -1;
72   }
73 
74   unsigned char *buf, *p;
75   struct stat st;
76   int bc_offset;
77 
78   fstat (fd, &st);
79   buf = (unsigned char *) mmap (NULL, st.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
80   close (fd);
81 
82   bc_offset = transferBytesToNumLe (buf+ 8, 4);
83   p = buf + 4 * 7;      // Offset to tag fields.
84 
85   while (p < buf + bc_offset) {
86     uint16_t tag, length;
87 
88     tag = transferBytesToNumLe (p, 2);
89     length = transferBytesToNumLe (p + 2, 2);
90     p += 4;
91     switch (tag) {
92       case 0x4002:       // Optimization Level,. e.g., 2 for -O2
93         mOptimizationLevel = transferBytesToNumLe (p, 4);
94         LOGV("Wrapper field: -O%d", mOptimizationLevel);
95         break;
96 
97       case 0x5002:       // LDFLAGS string
98         LOGV("Wrapper field: %s", p);
99         if (compiler.parseLDFlags (*this, reinterpret_cast<const char *>(p)) != 0) {
100           LOGE("Cannot parse ldflags from wrapper");
101           close(fd);
102           return -1;
103         }
104         break;
105 
106       case 0x4001:       // Compiler Version, e.g., 3300 for llvm-3.3.
107       case 0x5001:       // Bitcode Type, e.g., rel, shared, or exec.
108       default:
109         // Some field maybe useful, but we use wrapper to encode command line,
110         // this is not necessary for now.
111         break;
112     }
113 
114     p += (length + 3) & ~3;  // Data are always padding to 4-byte boundary.
115   }
116 
117   munmap (buf, st.st_size);
118   return 0;
119 }
120 
dropExternalLDLibs(SONameMap & map)121 void BitcodeInfo::dropExternalLDLibs(SONameMap &map) {
122   for (SONameMap::iterator i = map.begin(), e = map.end(); i != e; ++i) {
123     BitcodeInfo &info = i->second;
124     for (std::list<std::string>::iterator i_libs = info.mLDLibs.begin(),
125          e_libs = info.mLDLibs.end(); i_libs != e_libs; ) {
126       std::list<std::string>::iterator cur_libs = i_libs++;
127       std::string full_soname = std::string("lib") + *cur_libs + ".so";
128       if (map.find(full_soname) == map.end()) {
129         LOGV("Drop -l%s from %s for linking order decision", cur_libs->c_str(), info.mSOName.c_str());
130         info.mLDLibs.erase(cur_libs);
131       }
132     }
133   }
134 }
135 
136 // This function reads N bytes from BUFFER in little endian.
transferBytesToNumLe(const unsigned char * buffer,size_t n)137 int BitcodeInfo::transferBytesToNumLe(const unsigned char *buffer, size_t n) {
138   int ret = 0;
139   const unsigned char *p = buffer + n;
140 
141   while (--p >= buffer)
142     ret = ret * 0x100 + *p;
143 
144   return ret;
145 }
146 
BitcodeCompiler(const std::string & abi,const std::string & sysroot,const std::string & working_dir,const bool savetemps)147 BitcodeCompiler::BitcodeCompiler(const std::string &abi, const std::string &sysroot, const std::string &working_dir, const bool savetemps)
148   : mAbi(abi), mSysroot(sysroot), mWorkingDir(working_dir), mRet(RET_OK), mSaveTemps(savetemps) {
149   // CFlags
150   mGlobalCFlags = kGlobalTargetAttrs[mAbi].mBaseCFlags;
151   mGlobalCFlags += std::string(" -mtriple=") + kGlobalTargetAttrs[mAbi].mTriple;
152   mGlobalCFlags += " -filetype=obj -mc-relax-all";
153   mGlobalCFlags += " -relocation-model=pic -code-model=small -use-init-array";
154   mGlobalCFlags += " " OPTION_FUNCTION_SECTION;
155 
156   // LDFlags
157   mGlobalLDFlags = kGlobalTargetAttrs[mAbi].mBaseLDFlags;
158   mGlobalLDFlags += std::string(" -Bsymbolic -X -m ") + kGlobalTargetAttrs[mAbi].mLinkEmulation;
159   mGlobalLDFlags += std::string(" --sysroot=") + mSysroot;
160   mGlobalLDFlags += " --build-id --eh-frame-hdr";
161 
162   // LDLibs
163   mGlobalLDLibs = " ";
164 }
165 
~BitcodeCompiler()166 BitcodeCompiler::~BitcodeCompiler() {
167 }
168 
translate()169 void BitcodeCompiler::translate() {
170   for (std::vector<BitcodeInfo>::const_iterator i = mBitcodeFiles.begin(),
171        e = mBitcodeFiles.end(); i != e; ++i) {
172     const BitcodeInfo &bc = *i;
173     LOGV("Translate bitcode: %s -> %s", bc.mBCPath.c_str(), bc.mTargetBCPath.c_str());
174     std::string cmd = mExecutableToolsPath[(unsigned)CMD_TRANSLATE];
175     cmd += std::string(" -arch=") + kGlobalTargetAttrs[mAbi].mArch;
176     cmd += " " + bc.mBCPath + " -o " + bc.mTargetBCPath;
177     runCmd(cmd, /*dump=*/true);
178     if (returnCode() != RET_OK) {
179       mRet = RET_FAIL_TRANSLATE;
180       return;
181     }
182     if (!mSaveTemps)
183       removeIntermediateFile(bc.mBCPath);
184   }
185 }
186 
compile()187 void BitcodeCompiler::compile() {
188   for (std::vector<BitcodeInfo>::const_iterator i = mBitcodeFiles.begin(),
189        e = mBitcodeFiles.end(); i != e; ++i) {
190     const BitcodeInfo &bc = *i;
191     LOGV("Compile bitcode: %s -> %s", bc.mTargetBCPath.c_str(), bc.mObjPath.c_str());
192     std::ostringstream os;
193 
194     os << mExecutableToolsPath[(unsigned)CMD_COMPILE]
195        << " " << mGlobalCFlags
196        << " -O" << bc.mOptimizationLevel
197        << " " << bc.mTargetBCPath
198        << " -o " << bc.mObjPath;
199 
200     if (bc.mLDFlags.find("-pie") != std::string::npos)
201       os << " -enable-pie";
202 
203 #if ON_DEVICE && VERBOSE
204     Timer t_llc;
205     t_llc.start();
206 #endif
207     runCmd(os.str(), /*dump=*/true);
208 #if ON_DEVICE && VERBOSE
209     llc_usec += t_llc.stop();
210 #endif
211     if (returnCode() != RET_OK) {
212       mRet = RET_FAIL_COMPILE;
213       return;
214     }
215     if (!mSaveTemps)
216       removeIntermediateFile(bc.mTargetBCPath);
217   }
218 }
219 
link()220 void BitcodeCompiler::link() {
221   BitcodeInfo::dropExternalLDLibs(mSonameMap);
222 
223   while (!mSonameMap.empty()) {
224     SONameMap::iterator i = mSonameMap.begin(), e = mSonameMap.end();
225     for (; i != e; ++i) {
226       const BitcodeInfo &bc = i->second;
227 
228       if (bc.mLDLibs.empty()) {
229         // No internal dependency for this bitcode
230         LOGV("Link: %s -> %s", bc.mObjPath.c_str(), bc.mSOName.c_str());
231         std::string cmd = mExecutableToolsPath[(unsigned)CMD_LINK];
232         std::string libdir = (mAbi == TargetAbi::X86_64) ? "lib64" : "lib";
233         cmd += " " + mGlobalLDFlags;
234         cmd += " " + bc.mLDFlags;
235         if (bc.mShared) {
236           cmd += std::string(" ") + mSysroot + "/usr/" + libdir + "/crtbegin_so.o";
237           cmd += " -shared " + bc.mObjPath + " -o " + bc.mOutPath;
238           cmd += " -soname " + bc.mSOName;
239         } else if (bc.mStatic) {
240           cmd += std::string(" ") + mSysroot + "/usr/" + libdir + "/crtbegin_static.o";
241           cmd += " " + bc.mObjPath + " -o " + bc.mOutPath;
242         } else {
243           cmd += std::string(" ") + mSysroot + "/usr/" + libdir + "/crtbegin_dynamic.o";
244           cmd += " " + bc.mObjPath + " -o " + bc.mOutPath;
245         }
246         // Add ldlibs
247         cmd += " " + bc.mLDLocalLibsStr;
248         cmd += " " + mGlobalLDLibs;
249         cmd += " " + bc.mLDLibsStr;
250         cmd += " " + mExecutableToolsPath[(unsigned)CMD_LINK_RUNTIME];
251         if (!bc.mStatic)
252           cmd += std::string(" -ldl");
253         cmd += std::string(" -lc -lm"); // Libportable may uses libc symbols
254 
255         if (bc.mShared)
256           cmd += std::string(" ") + mSysroot + "/usr/" + libdir + "/crtend_so.o";
257         else
258           cmd += std::string(" ") + mSysroot + "/usr/" + libdir + "/crtend_android.o";
259         runCmd(cmd, /*dump=*/true);
260         if (returnCode() != RET_OK)
261           return;
262 
263         copyRuntime(bc);
264         if (!mSaveTemps)
265           removeIntermediateFile(bc.mObjPath);
266 
267         mSonameMap.erase(i);
268         BitcodeInfo::dropExternalLDLibs(mSonameMap);
269         break;  // Re-compute
270       }
271     } // for
272 
273     if (i == e) {
274       LOGE("Failed to compute linking order: Internal cyclic dependency!");
275       mRet = RET_FAIL_LINK;
276       return;
277     }
278   } // while
279 }
280 
runCmd(std::string cmd,bool dump)281 void BitcodeCompiler::runCmd(std::string cmd, bool dump) {
282   LOGV("Command: %s", cmd.c_str());
283   std::string logfilename = mWorkingDir + "/compile_log";
284   if (dump) {
285     cmd += " > " + logfilename + " 2>&1";
286   }
287   int ret = system(cmd.c_str());
288   if (ret != 0) {
289     mRet = RET_FAIL_RUN_CMD;
290     if (dump) {
291       std::ifstream ifs(logfilename.c_str());
292       std::stringstream sstr;
293       sstr << ifs.rdbuf();
294       LOGE("Error message: %s", sstr.str().c_str());
295       std::fstream fout;
296       std::string file = mWorkingDir + "/compile_error";
297       fout.open(file.c_str(), std::fstream::out | std::fstream::app);
298       fout << "Failed command: " << cmd << "\n";
299       fout << "Error message: " << sstr.str() << "\n";
300       fout.close();
301     }
302     return;
303   }
304   mRet = RET_OK;
305 }
306 
prepareBitcodes()307 void BitcodeCompiler::prepareBitcodes() {
308   getBitcodeFiles();
309   createSONameMapping();
310 }
311 
createSONameMapping()312 void BitcodeCompiler::createSONameMapping() {
313   for (std::vector<BitcodeInfo>::const_iterator i = mBitcodeFiles.begin(),
314        e = mBitcodeFiles.end(); i != e; ++i) {
315     const BitcodeInfo &info = *i;
316     LOGV("Map soname %s -> %s", info.mSOName.c_str(), info.mBCPath.c_str());
317     mSonameMap[info.mSOName] = info;
318   }
319 }
320