Embed RS metadata info directly in emitted object code.

Bug: 7419958
Change-Id: Ibd82e503c006532aa74013a46dd13ec3e9473daa
diff --git a/lib/Renderscript/RSEmbedInfo.cpp b/lib/Renderscript/RSEmbedInfo.cpp
new file mode 100644
index 0000000..1d68ee4
--- /dev/null
+++ b/lib/Renderscript/RSEmbedInfo.cpp
@@ -0,0 +1,149 @@
+/*
+ * Copyright 2012, The Android Open Source Project
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+#include "bcc/Assert.h"
+#include "bcc/Renderscript/RSTransforms.h"
+
+#include <cstdlib>
+
+#include <llvm/DerivedTypes.h>
+#include <llvm/Function.h>
+#include <llvm/Instructions.h>
+#include <llvm/IRBuilder.h>
+#include <llvm/Module.h>
+#include <llvm/Pass.h>
+#include <llvm/Support/raw_ostream.h>
+#include <llvm/Target/TargetData.h>
+#include <llvm/Type.h>
+
+#include "bcc/Config/Config.h"
+#include "bcc/Renderscript/RSInfo.h"
+#include "bcc/Support/Log.h"
+
+using namespace bcc;
+
+namespace {
+
+/* RSForEachExpandPass - This pass operates on functions that are able to be
+ * called via rsForEach() or "foreach_<NAME>". We create an inner loop for the
+ * ForEach-able function to be invoked over the appropriate data cells of the
+ * input/output allocations (adjusting other relevant parameters as we go). We
+ * support doing this for any ForEach-able compute kernels. The new function
+ * name is the original function name followed by ".expand". Note that we
+ * still generate code for the original function.
+ */
+class RSEmbedInfoPass : public llvm::ModulePass {
+private:
+  static char ID;
+
+  llvm::Module *M;
+  llvm::LLVMContext *C;
+
+  const RSInfo *mInfo;
+
+public:
+  RSEmbedInfoPass(const RSInfo *info)
+      : ModulePass(ID),
+        mInfo(info) {
+  }
+
+  virtual bool runOnModule(llvm::Module &M) {
+    this->M = &M;
+    C = &M.getContext();
+
+    std::string str;
+    llvm::raw_string_ostream s(str);
+
+    // We use a simple text format here that the compatibility library can
+    // easily parse. Each section starts out with its name followed by a count.
+    // The count denotes the number of lines to parse for that particular
+    // category. Variables and Functions merely put the appropriate identifier
+    // on the line, while ForEach kernels have the encoded int signature,
+    // followed by a hyphen followed by the identifier (function to look up).
+    // Object Slots are just listed as one integer per line.
+    const RSInfo::ExportVarNameListTy &export_vars = mInfo->getExportVarNames();
+    s << "exportVarCount: " << (unsigned int) export_vars.size() << "\n";
+    for (RSInfo::ExportVarNameListTy::const_iterator
+             export_var_iter = export_vars.begin(),
+             export_var_end = export_vars.end();
+         export_var_iter != export_var_end; export_var_iter++) {
+      s << *export_var_iter << "\n";
+    }
+
+    const RSInfo::ExportFuncNameListTy &export_funcs =
+        mInfo->getExportFuncNames();
+    s << "exportFuncCount: " << (unsigned int) export_funcs.size() << "\n";
+    for (RSInfo::ExportFuncNameListTy::const_iterator
+             export_func_iter = export_funcs.begin(),
+             export_func_end = export_funcs.end();
+         export_func_iter != export_func_end; export_func_iter++) {
+      s << *export_func_iter << "\n";
+    }
+
+    const RSInfo::ExportForeachFuncListTy &export_foreach_funcs =
+        mInfo->getExportForeachFuncs();
+    s << "exportForEachCount: "
+      << (unsigned int) export_foreach_funcs.size() << "\n";
+    for (RSInfo::ExportForeachFuncListTy::const_iterator
+             foreach_func_iter = export_foreach_funcs.begin(),
+             foreach_func_end = export_foreach_funcs.end();
+         foreach_func_iter != foreach_func_end; foreach_func_iter++) {
+      std::string name(foreach_func_iter->first);
+      s << foreach_func_iter->second << " - "
+        << foreach_func_iter->first << "\n";
+    }
+
+    const RSInfo::ObjectSlotListTy &object_slots = mInfo->getObjectSlots();
+    s << "objectSlotCount: " << (unsigned int) object_slots.size() << "\n";
+    for (RSInfo::ObjectSlotListTy::const_iterator
+             slots_iter = object_slots.begin(),
+             slots_end = object_slots.end();
+         slots_iter != slots_end; slots_iter++) {
+      s << slots_iter << "\n";
+    }
+
+    s.flush();
+
+    // Embed this as the global variable .rs.info so that it will be
+    // accessible from the shared object later.
+    llvm::Constant *Init = llvm::ConstantDataArray::getString(*C, str);
+    llvm::GlobalVariable *InfoGV =
+        new llvm::GlobalVariable(M, Init->getType(), true,
+                                 llvm::GlobalValue::ExternalLinkage, Init,
+                                 ".rs.info");
+    (void) InfoGV;
+
+    return true;
+  }
+
+  virtual const char *getPassName() const {
+    return "Embed Renderscript Info";
+  }
+
+};  // end RSEmbedInfoPass
+
+}  // end anonymous namespace
+
+char RSEmbedInfoPass::ID = 0;
+
+namespace bcc {
+
+llvm::ModulePass *
+createRSEmbedInfoPass(const RSInfo *info) {
+  return new RSEmbedInfoPass(info);
+}
+
+}  // end namespace bcc