2 * Copyright (c) 2016 Samsung Electronics Co., Ltd All Rights Reserved
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
8 * http://www.apache.org/licenses/LICENSE-2.0
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.
17 #include <pkgmgr-info.h>
18 #include <pkgmgr_installer_info.h>
23 #include "pkgmgr_parser_plugin_interface.h"
43 #define LOG_TAG "NETCORE_INSTALLER_PLUGIN"
45 #ifndef DEVICE_API_DIR
46 #error "DEVICE_API_DIR is missed"
50 #error "RUNTIME_DIR is missed"
54 #error "CROSSGEN_PATH is missed"
58 #define __STR(x) __XSTR(x)
59 static const char* __DEVICE_API_DIR = __STR(DEVICE_API_DIR);
60 static const char* __RUNTIME_DIR = __STR(RUNTIME_DIR);
61 static const char* __CROSSGEN_PATH = __STR(CROSSGEN_PATH);
62 static const char* __JIT_PATH = __STR(RUNTIME_DIR)"/libclrjit.so";
67 static std::string replace(std::string &str, const std::string& from, const std::string& to)
70 while ((startPos = str.find(from, startPos)) != std::string::npos) {
71 str.replace(startPos, from.length(), to);
72 startPos += to.length();
77 static void smack_(const char* dllPath, const char* label)
79 static const char* chsmack = "/usr/bin/chsmack";
86 waitpid(pid, &status, 0);
87 if (WIFEXITED(status))
90 const char* args[] = {
96 execv(chsmack, const_cast<char*const*>(args));
102 static void crossgen(const char* dllPath, const char* appPath)
104 //pid_t parent = getpid();
111 waitpid(pid, &status, 0);
112 if (WIFEXITED(status))
115 // search dlls in the application directory first, to use application dlls
116 // instead of system dlls when proceeding NI
117 std::vector<std::string> tpaDir;
118 if (appPath != NULL) {
119 std::string path(appPath);
120 std::string::size_type prevPos = 0, pos = 0;
121 while ((pos = path.find(':', pos)) != std::string::npos) {
122 std::string substring(path.substr(prevPos, pos - prevPos));
123 tpaDir.push_back(substring);
126 std::string substring(path.substr(prevPos, pos - prevPos));
127 tpaDir.push_back(substring);
129 tpaDir.push_back(__RUNTIME_DIR);
130 tpaDir.push_back(__DEVICE_API_DIR);
132 // get reference API directory ([DEVICE_API_DIR]/ref)
133 int len = strlen(__DEVICE_API_DIR);
134 char* refAPIDir = (char*)calloc(len + 4, 1);
136 printf("fail to allocate memory for reference API directory\n");
139 snprintf(refAPIDir, len + 4, "%s%s", __DEVICE_API_DIR, "/ref");
140 tpaDir.push_back(refAPIDir);
143 assembliesInDirectory(tpaDir, tpa);
145 std::vector<const char*> argv = {
147 "/Trusted_Platform_Assemblies", tpa.c_str(),
148 "/JITPath", __JIT_PATH,
149 "/FragileNonVersionable"
151 if (appPath != nullptr) {
152 argv.push_back("/App_Paths");
153 argv.push_back(appPath);
155 argv.push_back(dllPath);
156 argv.push_back(nullptr);
158 printf("+ %s\n", dllPath);
160 execv(__CROSSGEN_PATH, const_cast<char* const*>(argv.data()));
165 static int getRootPath(const char *pkgId, std::string& rootPath)
172 if (pkgmgr_installer_info_get_target_uid(&uid) < 0) {
173 _ERR("Failed to get UID");
177 _INFO("user id is %d", uid);
179 pkgmgrinfo_pkginfo_h handle;
181 ret = pkgmgrinfo_pkginfo_get_pkginfo(pkgId, &handle);
182 if (ret != PMINFO_R_OK)
185 ret = pkgmgrinfo_pkginfo_get_usr_pkginfo(pkgId, uid, &handle);
186 if (ret != PMINFO_R_OK)
190 ret = pkgmgrinfo_pkginfo_get_root_path(handle, &path);
191 if (ret != PMINFO_R_OK) {
192 pkgmgrinfo_pkginfo_destroy_pkginfo(handle);
196 pkgmgrinfo_pkginfo_destroy_pkginfo(handle);
201 static bool niExist(const std::string& path, std::string& ni)
203 // native image of System.Private.CoreLib.dll should have to overwrite
204 // original file to support new coreclr
205 if (path.find("System.Private.CoreLib.dll") != std::string::npos) {
206 std::string coreLibBackup = path + ".Backup";
207 if (!fileNotExist(coreLibBackup)) {
214 static const char* possibleExts[] = {
215 ".ni.dll", ".NI.dll", ".NI.DLL", ".ni.DLL",
216 ".ni.exe", ".NI.exe", ".NI.EXE", ".ni.EXE"
218 std::string fName = path.substr(0, path.size() - 4);
222 for (const char* ext : possibleExts) {
223 std::string f = fName + ext;
224 if (stat(f.c_str(), &sb) == 0) {
233 static void createCoreLibNI()
235 std::string coreLib = concatPath(__RUNTIME_DIR, "System.Private.CoreLib.dll");
236 std::string niCoreLib = concatPath(__RUNTIME_DIR, "System.Private.CoreLib.ni.dll");
237 std::string coreLibBackup = concatPath(__RUNTIME_DIR, "System.Private.CoreLib.dll.Backup");
239 if (!niExist(coreLib, niCoreLib)) {
240 crossgen(coreLib.c_str(), nullptr);
241 if (!fileNotExist(niCoreLib)) {
242 // change owner and groups for generated ni file.
244 if (!stat(coreLib.c_str(), &info)) {
245 if (chown(niCoreLib.c_str(), info.st_uid, info.st_gid) == -1)
246 _ERR("Failed to change owner and group name");
248 smack_(niCoreLib.c_str(), "_");
249 rename(coreLib.c_str(), coreLibBackup.c_str());
250 rename(niCoreLib.c_str(), coreLib.c_str());
255 void createNiPlatform()
259 const char* platformDirs[] = {__RUNTIME_DIR, __DEVICE_API_DIR, "/usr/bin"};
261 createNiUnderDirs(platformDirs, 3, [](const char* ni) {
266 void createNiSelect(const char* dllPath)
271 if (!fileNotExist(dllPath)) {
272 if (!niExist(dllPath, niPath)) {
273 crossgen(dllPath, nullptr);
274 if (niExist(dllPath, niPath)) {
275 // change owner and groups for generated ni file.
277 if (!stat(dllPath, &info)) {
278 if (chown(niPath.c_str(), info.st_uid, info.st_gid) == -1)
279 _ERR("Failed to change owner and group name");
281 smack_(niPath.c_str(), "_");
285 printf("Already [%s] file is exist\n", niPath.c_str());
289 void createNiUnderDirs(const char* rootPaths[], int count, const char* ignores[], int igcount, afterCreate cb)
291 std::string appPaths;
292 for (int i = 0; i < count; i++) {
293 appPaths += rootPaths[i];
297 if (appPaths.back() == ':')
300 auto convert = [&appPaths, ignores, igcount, &cb](const char* path, const char* name) {
301 for (int i = 0; i < igcount; i++) {
302 if (strcmp(path, ignores[i]) == 0)
306 if (isManagedAssembly(path) && !isNativeImage(path) && !niExist(path, ni)) {
307 crossgen(path, appPaths.c_str());
308 if (niExist(path, ni)) {
309 // change owner and groups for generated ni file.
311 if (!stat(path, &info)) {
312 if (chown(ni.c_str(), info.st_uid, info.st_gid) == -1)
313 _ERR("Failed to change owner and group name");
322 for (int i = 0; i < count; i++)
323 scanFilesInDir(rootPaths[i], convert, -1);
325 void createNiUnderDirs(const char* rootPaths[], int count, afterCreate cb)
327 createNiUnderDirs(rootPaths, count, nullptr, 0, cb);
329 void createNiUnderDirs(const char* rootPaths[], int count)
331 createNiUnderDirs(rootPaths, count, nullptr);
334 int createNiUnderPkgRoot(const char* pkgName)
337 if (getRootPath(pkgName, pkgRoot) < 0)
340 std::string binDir = concatPath(pkgRoot, "bin");
341 std::string libDir = concatPath(pkgRoot, "lib");
342 _INFO("bindir : %s", binDir.c_str());
343 _INFO("libdir : %s", libDir.c_str());
345 const char* paths[] = {
350 // change smack label for generated ni file.
351 std::string label = "User::Pkg::" + std::string(pkgName) + "::RO";
352 createNiUnderDirs(paths, 2, [label](const char* ni) {
353 smack_(ni, label.c_str());