63568b1430
Support any number of overlay packages. Support any target package. UPDATED PACKAGE MATCHING ------------------------ In Runtime resource overlay, iteration 1, only a single overlay package was considered. Package matching was based on file paths: /vendor/overlay/system/framework-res.apk corresponded to /system/framework-res.apk. Introduce a more flexible matching scheme where any package is an overlay package if its manifest includes <overlay targetPackage="com.target.package"/> For security reasons, an overlay package must fulfill certain criteria to take effect: see below. THE IDMAP TOOL AND IDMAP FILES ------------------------------ Idmap files are created by the 'idmap' binary; idmap files must be present when loading packages. For the Android system, Zygote calls 'idmap' as part of the resource pre-loading. For application packages, 'idmap' is invoked via 'installd' during package installation (similar to 'dexopt'). UPDATED FLOW ------------ The following is an outline of the start-up sequences for the Android system and Android apps. Steps marked with '+' are introduced by this commit. Zygote initialization Initial AssetManager object created + idmap --scan creates idmaps for overlays targeting 'android', \ stores list of overlays in /data/resource-cache/overlays.list AssetManager caches framework-res.apk + AssetManager caches overlay packages listed in overlays.list Android boot New AssetManager's ResTable acquired AssetManager re-uses cached framework-res.apk + AssetManager re-uses cached 'android' overlays (if any) App boot ActivityThread prepares AssetManager to load app.apk + ActivityThread prepares AssetManager to load app overlays (if any) New AssetManager's ResTable acquired as per Android boot SECURITY -------- Overlay packages are required to be pre-loaded (in /vendor/overlay). These packages are trusted by definition. A future iteration of runtime resource overlay may add support for downloaded overlays, which would likely require target and overlay signatures match for the overlay to be trusted. LOOKUP PRIORITY --------------- During resource lookup, packages are sequentially queried to provide a best match, given the constraints of the current configuration. If any package provide a better match than what has been found so far, it replaces the previous match. The target package is always queried last. When loading a package with more than one overlay, the order in which the overlays are added become significant if several packages overlay the same resource. Had downloaded overlays been supported, the install time could have been used to determine the load order. Regardless, for pre-installed overlays, the install time is randomly determined by the order in which the Package Manager locates the packages during initial boot. To support a well-defined order, pre-installed overlay packages are expected to define an additional 'priority' attribute in their <overlay> tags: <overlay targetPackage="com.target.package" priority="1234"/> Pre-installed overlays are loaded in order of their priority attributes, sorted in ascending order. Assigning the same priority to several overlays targeting the same base package leads to undefined behaviour. It is the responsibility of the vendor to avoid this. The following example shows the ResTable and PackageGroups after loading an application and two overlays. The resource lookup framework will query the packages in the order C, B, A. +------+------+- -+------+------+ | 0x01 | | ... | | 0x7f | +------+------+- -+------+------+ | | "android" Target package A | Pre-installed overlay B (priority 1) | Pre-installed overlay C (priority 2) Change-Id: If49c963149369b1957f7d2303b3dd27f669ed24e
214 lines
6.1 KiB
C
214 lines
6.1 KiB
C
/*
|
|
**
|
|
** Copyright 2008, 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.
|
|
*/
|
|
|
|
#define LOG_TAG "installd"
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <stdint.h>
|
|
#include <inttypes.h>
|
|
#include <sys/stat.h>
|
|
#include <dirent.h>
|
|
#include <unistd.h>
|
|
#include <ctype.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <utime.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/types.h>
|
|
#include <sys/wait.h>
|
|
|
|
#include <cutils/fs.h>
|
|
#include <cutils/sockets.h>
|
|
#include <cutils/log.h>
|
|
#include <cutils/properties.h>
|
|
#include <cutils/multiuser.h>
|
|
|
|
#include <private/android_filesystem_config.h>
|
|
|
|
#if INCLUDE_SYS_MOUNT_FOR_STATFS
|
|
#include <sys/mount.h>
|
|
#else
|
|
#include <sys/statfs.h>
|
|
#endif
|
|
|
|
#define SOCKET_PATH "installd"
|
|
|
|
|
|
/* elements combined with a valid package name to form paths */
|
|
|
|
#define PRIMARY_USER_PREFIX "data/"
|
|
#define SECONDARY_USER_PREFIX "user/"
|
|
|
|
#define PKG_DIR_POSTFIX ""
|
|
|
|
#define PKG_LIB_POSTFIX "/lib"
|
|
|
|
#define CACHE_DIR_POSTFIX "/cache"
|
|
|
|
#define APP_SUBDIR "app/" // sub-directory under ANDROID_DATA
|
|
|
|
#define APP_LIB_SUBDIR "app-lib/" // sub-directory under ANDROID_DATA
|
|
|
|
#define MEDIA_SUBDIR "media/" // sub-directory under ANDROID_DATA
|
|
|
|
/* other handy constants */
|
|
|
|
#define PRIVATE_APP_SUBDIR "app-private/" // sub-directory under ANDROID_DATA
|
|
|
|
#define DALVIK_CACHE_PREFIX "/data/dalvik-cache/"
|
|
#define DALVIK_CACHE_POSTFIX "/classes.dex"
|
|
|
|
#define UPDATE_COMMANDS_DIR_PREFIX "/system/etc/updatecmds/"
|
|
|
|
#define IDMAP_PREFIX "/data/resource-cache/"
|
|
#define IDMAP_SUFFIX "@idmap"
|
|
|
|
#define PKG_NAME_MAX 128 /* largest allowed package name */
|
|
#define PKG_PATH_MAX 256 /* max size of any path we use */
|
|
|
|
/* data structures */
|
|
|
|
typedef struct {
|
|
char* path;
|
|
size_t len;
|
|
} dir_rec_t;
|
|
|
|
typedef struct {
|
|
size_t count;
|
|
dir_rec_t* dirs;
|
|
} dir_rec_array_t;
|
|
|
|
extern dir_rec_t android_app_dir;
|
|
extern dir_rec_t android_app_private_dir;
|
|
extern dir_rec_t android_app_lib_dir;
|
|
extern dir_rec_t android_data_dir;
|
|
extern dir_rec_t android_asec_dir;
|
|
extern dir_rec_t android_media_dir;
|
|
extern dir_rec_array_t android_system_dirs;
|
|
|
|
typedef struct cache_dir_struct {
|
|
struct cache_dir_struct* parent;
|
|
int32_t childCount;
|
|
int32_t hiddenCount;
|
|
int32_t deleted;
|
|
char name[];
|
|
} cache_dir_t;
|
|
|
|
typedef struct {
|
|
cache_dir_t* dir;
|
|
time_t modTime;
|
|
char name[];
|
|
} cache_file_t;
|
|
|
|
typedef struct {
|
|
size_t numDirs;
|
|
size_t availDirs;
|
|
cache_dir_t** dirs;
|
|
size_t numFiles;
|
|
size_t availFiles;
|
|
cache_file_t** files;
|
|
size_t numCollected;
|
|
void* memBlocks;
|
|
int8_t* curMemBlockAvail;
|
|
int8_t* curMemBlockEnd;
|
|
} cache_t;
|
|
|
|
/* util.c */
|
|
|
|
int create_pkg_path_in_dir(char path[PKG_PATH_MAX],
|
|
const dir_rec_t* dir,
|
|
const char* pkgname,
|
|
const char* postfix);
|
|
|
|
int create_pkg_path(char path[PKG_PATH_MAX],
|
|
const char *pkgname,
|
|
const char *postfix,
|
|
userid_t userid);
|
|
|
|
int create_user_path(char path[PKG_PATH_MAX],
|
|
userid_t userid);
|
|
|
|
int create_user_media_path(char path[PKG_PATH_MAX], userid_t userid);
|
|
|
|
int create_move_path(char path[PKG_PATH_MAX],
|
|
const char* pkgname,
|
|
const char* leaf,
|
|
userid_t userid);
|
|
|
|
int is_valid_package_name(const char* pkgname);
|
|
|
|
int create_cache_path(char path[PKG_PATH_MAX], const char *src);
|
|
|
|
int delete_dir_contents(const char *pathname,
|
|
int also_delete_dir,
|
|
const char *ignore);
|
|
|
|
int delete_dir_contents_fd(int dfd, const char *name);
|
|
|
|
int lookup_media_dir(char basepath[PATH_MAX], const char *dir);
|
|
|
|
int64_t data_disk_free();
|
|
|
|
cache_t* start_cache_collection();
|
|
|
|
void add_cache_files(cache_t* cache, const char *basepath, const char *cachedir);
|
|
|
|
void clear_cache_files(cache_t* cache, int64_t free_size);
|
|
|
|
void finish_cache_collection(cache_t* cache);
|
|
|
|
int validate_system_app_path(const char* path);
|
|
|
|
int get_path_from_env(dir_rec_t* rec, const char* var);
|
|
|
|
int get_path_from_string(dir_rec_t* rec, const char* path);
|
|
|
|
int copy_and_append(dir_rec_t* dst, const dir_rec_t* src, const char* suffix);
|
|
|
|
int validate_apk_path(const char *path);
|
|
|
|
int append_and_increment(char** dst, const char* src, size_t* dst_size);
|
|
|
|
char *build_string2(char *s1, char *s2);
|
|
char *build_string3(char *s1, char *s2, char *s3);
|
|
|
|
int ensure_dir(const char* path, mode_t mode, uid_t uid, gid_t gid);
|
|
int ensure_media_user_dirs(userid_t userid);
|
|
|
|
/* commands.c */
|
|
|
|
int install(const char *pkgname, uid_t uid, gid_t gid, const char *seinfo);
|
|
int uninstall(const char *pkgname, userid_t userid);
|
|
int renamepkg(const char *oldpkgname, const char *newpkgname);
|
|
int fix_uid(const char *pkgname, uid_t uid, gid_t gid);
|
|
int delete_user_data(const char *pkgname, userid_t userid);
|
|
int make_user_data(const char *pkgname, uid_t uid, userid_t userid, const char* seinfo);
|
|
int delete_user(userid_t userid);
|
|
int delete_cache(const char *pkgname, userid_t userid);
|
|
int move_dex(const char *src, const char *dst);
|
|
int rm_dex(const char *path);
|
|
int protect(char *pkgname, gid_t gid);
|
|
int get_size(const char *pkgname, userid_t userid, const char *apkpath, const char *libdirpath,
|
|
const char *fwdlock_apkpath, const char *asecpath, int64_t *codesize,
|
|
int64_t *datasize, int64_t *cachesize, int64_t *asecsize);
|
|
int free_cache(int64_t free_size);
|
|
int dexopt(const char *apk_path, uid_t uid, int is_public);
|
|
int movefiles();
|
|
int linklib(const char* target, const char* source, int userId);
|
|
int idmap(const char *target_path, const char *overlay_path, uid_t uid);
|