| /* |
| * Copyright (C) 2000-2009, Parallels, Inc. All rights reserved. |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, write to the Free Software |
| * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| */ |
| #ifndef _GNU_SOURCE |
| #define _GNU_SOURCE |
| #endif |
| #include <stdlib.h> |
| #include <stdio.h> |
| #include <sys/types.h> |
| #include <sys/stat.h> |
| #include <unistd.h> |
| #include <string.h> |
| #include <errno.h> |
| #include <ctype.h> |
| #include <sys/mman.h> |
| #include <fcntl.h> |
| #include <arpa/inet.h> |
| #include <stdarg.h> |
| #include <limits.h> |
| #include <dirent.h> |
| #include <sys/utsname.h> |
| |
| #include "util.h" |
| #include "logger.h" |
| #include "fs.h" |
| |
| #ifndef NR_OPEN |
| #define NR_OPEN 1024 |
| #endif |
| |
| char *unescapestr(char *src) |
| { |
| char *p1, *p2; |
| int fl; |
| |
| if (src == NULL) |
| return NULL; |
| p2 = src; |
| p1 = p2; |
| fl = 0; |
| while (*p2) { |
| if (*p2 == '\\' && !fl) { |
| fl = 1; |
| p2++; |
| } else { |
| *p1 = *p2; |
| p1++; p2++; |
| fl = 0; |
| } |
| } |
| *p1 = 0; |
| |
| return src; |
| } |
| |
| char *parse_line(char *str, char *ltoken, int lsz) |
| { |
| char *sp = str; |
| char *ep, *p; |
| int len; |
| |
| unescapestr(str); |
| while (*sp && isspace(*sp)) sp++; |
| if (!*sp || *sp == '#') |
| return NULL; |
| ep = sp + strlen(sp) - 1; |
| while (isspace(*ep) && ep >= sp) *ep-- = '\0'; |
| if (*ep == '"') |
| *ep = 0; |
| if (!(p = strchr(sp, '='))) |
| return NULL; |
| len = p - sp; |
| if (len >= lsz) |
| return NULL; |
| strncpy(ltoken, sp, len); |
| ltoken[len] = 0; |
| if (*(++p) == '"') |
| p++; |
| |
| return p; |
| } |
| |
| /* |
| 1 - exist |
| 0 - does't exist |
| -1 - error |
| */ |
| int stat_file(const char *file) |
| { |
| struct stat st; |
| |
| if (stat(file, &st)) { |
| if (errno != ENOENT) |
| return -1; |
| return 0; |
| } |
| return 1; |
| } |
| |
| int make_dir(char *path, int full) |
| { |
| char buf[4096]; |
| char *ps, *p; |
| int len; |
| |
| if (path == NULL) |
| return 0; |
| |
| ps = path + 1; |
| while ((p = strchr(ps, '/'))) { |
| len = p - path + 1; |
| snprintf(buf, len, "%s", path); |
| ps = p + 1; |
| if (!stat_file(buf)) { |
| if (mkdir(buf, 0755)) { |
| logger(-1, errno, "Can't create directory %s", |
| buf); |
| return 1; |
| } |
| } |
| } |
| if (!full) |
| return 0; |
| if (!stat_file(path)) { |
| if (mkdir(path, 0755)) { |
| logger(-1, errno, "Can't create directory %s", path); |
| return 1; |
| } |
| } |
| return 0; |
| } |
| |
| int parse_int(const char *str, int *val) |
| { |
| char *tail; |
| |
| errno = 0; |
| *val = (int)strtol(str, (char **)&tail, 10); |
| if (*tail != '\0' || errno == ERANGE) |
| return 1; |
| return 0; |
| } |
| |
| int parse_ul(const char *str, unsigned long *val) |
| { |
| char *tail; |
| |
| if (!strcmp(str, "unlimited")) { |
| *val = LONG_MAX; |
| return 0; |
| } |
| |
| errno = 0; |
| *val = (int)strtoul(str, (char **)&tail, 10); |
| if (*tail != '\0' || errno == ERANGE) |
| return ERR_INVAL; |
| return 0; |
| } |
| |
| void str_tolower(const char *from, char *to) |
| { |
| if (from == NULL || to == NULL) |
| return; |
| while ((*to++ = tolower(*from++))); |
| } |
| |
| void str_toupper(const char *from, char *to) |
| { |
| if (from == NULL || to == NULL) |
| return; |
| while ((*to++ = toupper(*from++))); |
| } |
| |
| int check_var(const void *val, const char *message) |
| { |
| if (val != NULL) |
| return 0; |
| logger(-1, 0, "%s", message); |
| |
| return 1; |
| } |
| |
| int cp_file(char *dst, char *src) |
| { |
| int fd_src, fd_dst, len, ret = 0; |
| struct stat st; |
| char buf[4096]; |
| |
| if (stat(src, &st) < 0) { |
| logger(-1, errno, "Unable to stat %s", src); |
| return -1; |
| } |
| len = st.st_size; |
| if ((fd_src = open(src, O_RDONLY)) < 0) { |
| logger(-1, errno, "Unable to open %s", src); |
| return -1; |
| } |
| if ((fd_dst = open(dst, O_CREAT|O_RDWR, st.st_mode)) < 0) { |
| logger(-1, errno, "Unable to open %s", dst); |
| close(fd_src); |
| return -1; |
| } |
| while(1) { |
| ret = read(fd_src, buf, sizeof(buf)); |
| if (!ret) |
| break; |
| else if (ret < 0) { |
| logger(-1, errno, "Unable to read from %s", src); |
| ret = -1; |
| break; |
| } |
| if (write(fd_dst, buf, ret) < 0) { |
| logger(-1, errno, "Unable to write to %s", dst); |
| ret = -1; |
| break; |
| } |
| } |
| close(fd_src); |
| close(fd_dst); |
| |
| return ret; |
| } |
| |
| void get_vps_conf_path(envid_t veid, char *buf, int len) |
| { |
| snprintf(buf, len, VPS_CONF_DIR "%d.conf", veid); |
| } |
| |
| char *arg2str(char **arg) |
| { |
| char **p; |
| char *str, *sp; |
| int len = 0; |
| |
| if (arg == NULL) |
| return NULL; |
| p = arg; |
| while (*p) |
| len += strlen(*p++) + 1; |
| if ((str = (char *)malloc(len + 1)) == NULL) |
| return NULL; |
| p = arg; |
| sp = str; |
| while (*p) |
| sp += sprintf(sp, "%s ", *p++); |
| |
| return str; |
| } |
| |
| void free_arg(char **arg) |
| { |
| while (*arg) free(*arg++); |
| } |
| |
| inline unsigned long max_ul(unsigned long val1, unsigned long val2) |
| { |
| return (val1 > val2) ? val1 : val2; |
| } |
| |
| inline unsigned long min_ul(unsigned long val1, unsigned long val2) |
| { |
| return (val1 < val2) ? val1 : val2; |
| } |
| |
| int yesno2id(const char *str) |
| { |
| if (!strcmp(str, "yes")) |
| return YES; |
| else if (!strcmp(str, "no")) |
| return NO; |
| return -1; |
| } |
| |
| int get_netaddr(const char *ip_str, void *ip) |
| { |
| if (strchr(ip_str, ':')) { |
| if (inet_pton(AF_INET6, ip_str, ip) <= 0) |
| return -1; |
| return AF_INET6; |
| } |
| if (inet_pton(AF_INET, ip_str, ip) <= 0) |
| return -1; |
| return AF_INET; |
| } |
| |
| const char *get_netname(unsigned int *ip, int family) |
| { |
| static char buf[INET6_ADDRSTRLEN]; |
| |
| return inet_ntop(family, ip, buf, sizeof(buf)); |
| } |
| |
| char *subst_VEID(envid_t veid, char *src) |
| { |
| char *srcp; |
| char str[STR_SIZE]; |
| char *sp, *se; |
| int r; |
| unsigned int len, veidlen; |
| |
| if (src == NULL) |
| return NULL; |
| /* Skip end '/' */ |
| se = src + strlen(src) - 1; |
| while (se != str && *se == '/') { |
| *se = 0; |
| se--; |
| } |
| if ((srcp = strstr(src, "$VEID"))) |
| veidlen = sizeof("$VEID") - 1; |
| else if ((srcp = strstr(src, "${VEID}"))) |
| veidlen = sizeof("${VEID}") - 1; |
| else |
| return strdup(src); |
| |
| sp = str; |
| se = str + sizeof(str); |
| len = srcp - src; /* Length of src before $VEID */ |
| if (len > sizeof(str)) |
| return NULL; |
| memcpy(str, src, len); |
| sp += len; |
| r = snprintf(sp, se - sp, "%d", veid); |
| sp += r; |
| if ((r < 0) || (sp >= se)) |
| return NULL; |
| if (*srcp) { |
| r = snprintf(sp, se - sp, "%s", srcp + veidlen); |
| sp += r; |
| if ((r < 0) || (sp >= se)) |
| return NULL; |
| } |
| return strdup(str); |
| } |
| |
| int get_pagesize() |
| { |
| long pagesize; |
| |
| if ((pagesize = sysconf(_SC_PAGESIZE)) == -1) { |
| logger(-1, errno, "Unable to get page size"); |
| return -1; |
| } |
| return pagesize; |
| } |
| |
| int get_mem(unsigned long long *mem) |
| { |
| long pages, pagesize; |
| if ((pages = sysconf(_SC_PHYS_PAGES)) == -1) { |
| logger(-1, errno, "Unable to get total phys pages"); |
| return -1; |
| } |
| if ((pagesize = get_pagesize()) < 0) |
| return -1; |
| *mem = (unsigned long long) pages * pagesize; |
| return 0; |
| } |
| |
| int get_thrmax(int *thrmax) |
| { |
| FILE *fd; |
| char str[128]; |
| |
| if (thrmax == NULL) |
| return 1; |
| if ((fd = fopen(PROCTHR, "r")) == NULL) { |
| logger(-1, errno, "Unable to open " PROCTHR); |
| return 1; |
| } |
| if (fgets(str, sizeof(str), fd) == NULL) { |
| fclose(fd); |
| return 1; |
| } |
| fclose(fd); |
| if (sscanf(str, "%du", thrmax) != 1) |
| return 1; |
| return 0; |
| } |
| |
| int get_swap(unsigned long long *swap) |
| { |
| FILE *fd; |
| char str[128]; |
| |
| if ((fd = fopen(PROCMEM, "r")) == NULL) { |
| logger(-1, errno, "Cannot open " PROCMEM); |
| return -1; |
| } |
| while (fgets(str, sizeof(str), fd)) { |
| if (sscanf(str, "SwapTotal: %llu", swap) == 1) { |
| *swap *= 1024; |
| fclose(fd); |
| return 0; |
| } |
| } |
| logger(-1, errno, "Swap: is not found in " PROCMEM ); |
| fclose(fd); |
| |
| return -1; |
| } |
| |
| int get_num_cpu() |
| { |
| FILE *fd; |
| char str[128]; |
| int ncpu = 0; |
| |
| if ((fd = fopen("/proc/cpuinfo", "r")) == NULL) { |
| logger(-1, errno, "Cannot open /proc/cpuinfo"); |
| return 1; |
| } |
| while (fgets(str, sizeof(str), fd)) { |
| if (!strncmp(str, "processor", 9)) |
| ncpu++; |
| } |
| fclose(fd); |
| return !ncpu ? 1 : ncpu; |
| } |
| |
| int get_lowmem(unsigned long long *mem) |
| { |
| FILE *fd; |
| char str[128]; |
| |
| if ((fd = fopen(PROCMEM, "r")) == NULL) { |
| logger(-1, errno, "Cannot open " PROCMEM); |
| return -1; |
| } |
| *mem = 0; |
| while (fgets(str, sizeof(str), fd)) { |
| if (sscanf(str, "LowTotal: %llu", mem) == 1) |
| break; |
| /* Use MemTotal in case LowTotal not found */ |
| sscanf(str, "MemTotal: %llu", mem); |
| } |
| fclose(fd); |
| if (*mem == 0) { |
| fprintf(stderr, "Neither LowTotal nor MemTotal found in the " |
| PROCMEM "\n"); |
| return -1; |
| } |
| *mem *= 1024; |
| return 0; |
| } |
| |
| char *get_file_name(char *str) |
| { |
| char *p; |
| int len; |
| |
| len = strlen(str) - sizeof(".conf") + 1; |
| if (len <= 0) |
| return NULL; |
| if (strcmp(str + len, ".conf")) |
| return NULL; |
| if ((p = malloc(len + 1)) == NULL) |
| return NULL; |
| strncpy(p, str, len); |
| p[len] = 0; |
| |
| return p; |
| } |
| |
| const char *get_vps_state_str(int vps_state) |
| { |
| const char *p = NULL; |
| |
| switch (vps_state) { |
| case STATE_RUNNING: |
| p = "running"; |
| break; |
| case STATE_STARTING: |
| p = "starting"; |
| break; |
| case STATE_STOPPED: |
| p = "stopped"; |
| break; |
| case STATE_STOPPING: |
| p = "stopping"; |
| break; |
| case STATE_RESTORING: |
| p = "restoring"; |
| break; |
| case STATE_CHECKPOINTING: |
| p = "checkpointing"; |
| break; |
| } |
| return p; |
| } |
| |
| int get_dump_file(unsigned veid, const char *dumpdir, char *buf, int size) |
| { |
| return snprintf(buf, size, "%s/" DEF_DUMPFILE, |
| dumpdir != NULL ? dumpdir : DEF_DUMPDIR, veid); |
| } |
| |
| int set_not_blk(int fd) |
| { |
| int oldfl, ret; |
| |
| if ((oldfl = fcntl(fd, F_GETFL)) == -1) |
| return -1; |
| oldfl |= O_NONBLOCK; |
| ret = fcntl(fd, F_SETFL, oldfl); |
| |
| return ret; |
| } |
| |
| /** Close all fd. |
| * @param close_std flag for closing the [0-2] fds |
| * @param ... list of fds are skiped, (-1 is the end mark) |
| */ |
| void close_fds(int close_std, ...) |
| { |
| int fd, max; |
| unsigned int i; |
| va_list ap; |
| int skip_fds[255]; |
| |
| max = sysconf(_SC_OPEN_MAX); |
| if (max < NR_OPEN) |
| max = NR_OPEN; |
| if (close_std) { |
| fd = open("/dev/null", O_RDWR); |
| if (fd != -1) { |
| dup2(fd, 0); dup2(fd, 1); dup2(fd, 2); |
| } else { |
| close(0); close(1); close(2); |
| } |
| } |
| /* build aray of skiped fds */ |
| va_start(ap, close_std); |
| skip_fds[0] = -1; |
| for (i = 0; i < (sizeof(skip_fds)/sizeof(skip_fds[0])); i++) { |
| fd = va_arg(ap, int); |
| skip_fds[i] = fd; |
| if (fd == -1) |
| break; |
| } |
| va_end(ap); |
| for (fd = 3; fd < max; fd++) { |
| for (i = 0; skip_fds[i] != fd && skip_fds[i] != -1; i++); |
| if (skip_fds[i] == fd) |
| continue; |
| close(fd); |
| } |
| } |
| |
| static void __move_config(int veid, int action, const char *prefix) |
| { |
| char conf[PATH_LEN]; |
| char newconf[PATH_LEN]; |
| |
| snprintf(conf, sizeof(conf), VPS_CONF_DIR "%d.%s", veid, prefix); |
| snprintf(newconf, sizeof(newconf), "%s." DESTR_PREFIX, conf); |
| action == BACKUP ? rename(conf, newconf) : unlink(newconf); |
| } |
| |
| /* Renames or removes CT config and various CT scripts. |
| */ |
| int move_config(int veid, int action) |
| { |
| __move_config(veid, action, "conf"); |
| __move_config(veid, action, MOUNT_PREFIX); |
| __move_config(veid, action, UMOUNT_PREFIX); |
| __move_config(veid, action, PRE_MOUNT_PREFIX); |
| __move_config(veid, action, POST_UMOUNT_PREFIX); |
| __move_config(veid, action, START_PREFIX); |
| __move_config(veid, action, STOP_PREFIX); |
| |
| return 0; |
| } |
| |
| void remove_names(envid_t veid) |
| { |
| char buf[STR_SIZE]; |
| char content[STR_SIZE]; |
| struct stat st; |
| struct dirent *ep; |
| DIR *dp; |
| char *p; |
| int r; |
| envid_t id; |
| |
| if (!(dp = opendir(VENAME_DIR))) |
| return; |
| while ((ep = readdir(dp))) { |
| snprintf(buf, sizeof(buf), VENAME_DIR "/%s", ep->d_name); |
| if (lstat(buf, &st)) |
| continue; |
| if (!S_ISLNK(st.st_mode)) |
| continue; |
| r = readlink(buf, content, sizeof(content) - 1); |
| if (r < 0) |
| continue; |
| content[r] = 0; |
| if ((p = strrchr(content, '/')) != NULL) |
| p++; |
| if (sscanf(p, "%d.conf", &id) == 1 && veid == id) |
| unlink(buf); |
| } |
| closedir(dp); |
| } |
| |
| size_t vz_strlcat(char *dst, const char *src, size_t count) |
| { |
| size_t dsize = strlen(dst); |
| size_t len = strlen(src); |
| size_t res = dsize + len; |
| |
| if (dsize >= count) |
| return dsize; |
| |
| dst += dsize; |
| count -= dsize; |
| if (len >= count) |
| len = count - 1; |
| |
| memcpy(dst, src, len); |
| dst[len] = 0; |
| |
| return res; |
| } |
| |
| #define MAX_OSREL_LEN 128 |
| |
| static void read_osrelease_conf(const char *dist, char *osrelease) |
| { |
| FILE *f; |
| char str[MAX_OSREL_LEN]; |
| char var[MAX_OSREL_LEN]; |
| char value[MAX_OSREL_LEN]; |
| int dlen = strlen(dist); |
| |
| if ((f = fopen(OSRELEASE_CFG, "r")) == NULL) { |
| logger(-1, errno, "Can't open file " OSRELEASE_CFG); |
| return; |
| } |
| while (fgets(str, sizeof(str) - 1, f) != NULL) { |
| if (str[0] == '#') |
| continue; |
| if (sscanf(str, " %s %s ", var, value) != 2) |
| continue; |
| if (strncmp(var, dist, strnlen(var, dlen)) == 0) { |
| strcpy(osrelease, value); |
| break; |
| } |
| } |
| fclose(f); |
| return; |
| } |
| |
| #define KVER(a,b,c) (((a) << 16) + ((b) << 8) + (c)) |
| int compare_osrelease(const char *cur, const char *min) |
| { |
| int cur_a, cur_b, cur_c; |
| int min_a, min_b, min_c; |
| int ret; |
| |
| ret = sscanf(cur, "%d.%d.%d", &cur_a, &cur_b, &cur_c); |
| if (ret != 3) { |
| logger(-1, 0, "Unable to parse kernel osrelease (%s)", cur); |
| return -1; |
| } |
| |
| ret = sscanf(min, "%d.%d.%d", &min_a, &min_b, &min_c); |
| if (ret != 3) { |
| logger(-1, 0, "Unable to parse value (%s) from " |
| OSRELEASE_CFG, min); |
| return -1; |
| } |
| |
| if (KVER(cur_a, cur_b, cur_c) < KVER(min_a, min_b, min_c)) |
| return 1; /* Current version is too old */ |
| |
| return 0; |
| } |
| #undef KVER |
| |
| /** Find out if a container needs setting osrelease, |
| * and set it if needed. */ |
| void get_osrelease(vps_res *res) |
| { |
| const char *dist; |
| char osrelease[MAX_OSREL_LEN] = ""; |
| struct utsname uts; |
| char *suffix; |
| int len; |
| |
| dist = get_dist_name(&res->tmpl); |
| if (!dist) |
| return; |
| |
| read_osrelease_conf(dist, osrelease); |
| if (osrelease[0] == '\0') |
| return; |
| |
| logger(1, 0, "Found osrelease %s for dist %s", osrelease, dist); |
| |
| /* Check if current osrelease is sufficient */ |
| if (uname(&uts) != 0) { |
| logger(-1, errno, "Error in uname()"); |
| return; |
| } |
| |
| if (compare_osrelease(uts.release, osrelease) < 1) |
| /* -1: error; 0: current version is good enough */ |
| return; |
| |
| /* Yes we need to set osrelease for this container */ |
| |
| /* Make version look like our kernel, i.e. add suffix |
| * like -028stab078.10 to osrelease |
| */ |
| if ((suffix = strchr(uts.release, '-')) != NULL) { |
| len = sizeof(osrelease) - strlen(osrelease); |
| strncat(osrelease, suffix, len); |
| osrelease[sizeof(osrelease) - 1] = 0; |
| } |
| |
| logger(1, 0, "Set osrelease=%s", osrelease); |
| res->env.osrelease = strdup(osrelease); |
| } |
| |
| static int envid_sort_fn(const void *val1, const void *val2) |
| { |
| const envid_t *r1 = (const envid_t *)val1; |
| const envid_t *r2 = (const envid_t *)val2; |
| |
| return (*r1 - *r2); |
| } |
| |
| /** Returns a sorted array of all running CTs. |
| * Caller needs to free() it after use |
| */ |
| int get_running_ve_list(envid_t **ves) |
| { |
| FILE *fp; |
| int res; |
| envid_t veid; |
| int venum = 0; |
| int ves_size = 256; |
| |
| |
| *ves = malloc(ves_size * sizeof(envid_t)); |
| if (*ves == NULL) |
| return -ENOMEM; |
| |
| if ((fp = fopen(PROCVEINFO, "r")) == NULL) { |
| return -errno; |
| } |
| while (!feof(fp)) { |
| res = fscanf(fp, "%d %*[^\n]", &veid); |
| if (res != 1 || !veid) |
| continue; |
| if (venum >= ves_size) |
| ves_size *= 2; |
| *ves = realloc(*ves, ves_size * sizeof(envid_t)); |
| if (*ves == NULL) |
| { |
| venum=-ENOMEM; |
| goto out; |
| } |
| (*ves)[venum++] = veid; |
| } |
| qsort(*ves, venum, sizeof(envid_t), envid_sort_fn); |
| out: |
| fclose(fp); |
| return venum; |
| } |
| |
| /* Searches for ve in velist */ |
| int ve_in_list(envid_t *list, int size, envid_t ve) |
| { |
| return bsearch(&ve, list, size, sizeof(envid_t), |
| envid_sort_fn) != NULL; |
| } |