blob: f6f5915dab4aae092d886e9cb16693773b844760 [file] [log] [blame] [raw]
/*
* 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
*/
#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 "util.h"
#include "logger.h"
#include "fs.h"
#ifndef NR_OPEN
#define NR_OPEN 1024
#endif
static const char *unescapestr(char *const src)
{
char *p1, *p2;
int fl;
if (src == NULL)
return NULL;
p1 = p2 = src;
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 **err)
{
char *sp = str;
char *ep, *p, *ret;
int len;
*err = NULL;
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 (!(p = strchr(sp, '='))) {
*err = "'=' not found";
return NULL;
}
len = p - sp;
if (len >= lsz) {
*err = "too long value";
return NULL;
}
strncpy(ltoken, sp, len);
ltoken[len] = 0;
if (*(++p) != '"')
return p;
/* Quoted argument requires some additional processing */
ret = ++p; /* skip opening quote */
/* Find the matching closing quote */
if (!(p = strrchr(p, '"'))) {
*err="unmatched quotes";
return NULL;
}
*p = '\0';
return ret;
}
/** Check if given file exists
* Returns:
* 1 - file exists
* 0 - file does not exist
* -1 - some system error
*/
int stat_file(const char *file)
{
if (access(file, F_OK) == 0)
return 1;
if (errno == ENOENT)
return 0;
logger(-1, errno, "Can't access file %s", file);
return -1;
}
/** Check if a directory is empty
* Returns:
* 1 - empty or nonexistent
* 0 - not empty
* -1 - error
*/
int dir_empty(const char *dir)
{
DIR *dp;
struct dirent *ep;
int ret = 1;
dp = opendir(dir);
if (dp == NULL) {
if (errno == ENOENT)
return 1;
logger(-1, errno, "Can't opendir %s", dir);
return -1;
}
while ((ep = readdir(dp))) {
if (!strcmp(ep->d_name, "."))
continue;
if (!strcmp(ep->d_name, ".."))
continue;
/* Not empty */
ret = 0;
break;
}
closedir(dp);
return ret;
}
int make_dir_mode(const char *path, int full, int mode)
{
char buf[4096];
const 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) != 1) {
if (mkdir(buf, mode)) {
logger(-1, errno, "Can't create directory %s",
buf);
return 1;
}
}
}
if (!full)
return 0;
if (stat_file(path) != 1) {
if (mkdir(path, mode)) {
logger(-1, errno, "Can't create directory %s", path);
return 1;
}
}
return 0;
}
int make_dir(const char *path, int full)
{
return make_dir_mode(path, full, 0755);
}
char *get_fs_root(const char *dir)
{
struct stat st;
dev_t id;
size_t len;
const char *p, *prev;
char tmp[PATH_MAX];
if (stat(dir, &st) < 0)
return NULL;
id = st.st_dev;
len = strlen(dir);
if (len > sizeof(tmp) - 1) {
errno = ERANGE;
return NULL;
}
p = dir + len;
prev = p;
while (p > dir) {
while (p > dir && *p == '/') p--;
while (p > dir && *p != '/') p--;
if (p <= dir)
break;
len = p - dir + 1;
strncpy(tmp, dir, len);
tmp[len] = 0;
if (stat(tmp, &st) < 0)
return NULL;
if (id != st.st_dev)
break;
prev = p;
}
len = prev - dir;
if (len) {
strncpy(tmp, dir, len);
tmp[len] = 0;
return strdup(tmp);
}
return NULL;
}
int parse_int(const char *str, int *val)
{
char *tail;
long res;
res = strtol(str, &tail, 10);
if (*tail != '\0' || res < INT_MIN || res > INT_MAX)
return 1;
*val = (int)res;
return 0;
}
int parse_ul(const char *str, unsigned long *val)
{
char *tail;
unsigned long res;
if (!strcmp(str, "unlimited")) {
*val = LONG_MAX;
return 0;
}
res = strtoul(str, &tail, 10);
if (*tail != '\0' || res > LONG_MAX)
return ERR_INVAL;
*val = res;
return 0;
}
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, ret = 0;
struct stat st;
char buf[4096];
if (stat(src, &st) < 0) {
logger(-1, errno, "Unable to stat %s", src);
return -1;
}
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_TRUNC|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, VPSCONFDIR "/%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_addr_family(const char *addr)
{
if (strchr(addr, ':'))
return AF_INET6;
else
return AF_INET;
}
int get_netaddr(const char *ip_str, void *ip)
{
int family;
family = get_addr_family(ip_str);
if (inet_pton(family, ip_str, ip) <= 0)
return -1;
return family;
}
static inline int max_netmask(int family)
{
if (family == AF_INET)
return 32;
else if (family == AF_INET6)
return 128;
else
return -1;
}
/* Check and "canonicalize" an IP address with optional netmask
* (in CIDR notation). Basically */
char *canon_ip(const char *str)
{
const char *ipstr, *maskstr;
int mask, family;
unsigned int ip[4];
static char dst[INET6_ADDRSTRLEN + sizeof("/128")];
maskstr = strchr(str, '/');
if (maskstr) {
ipstr = strndupa(str, maskstr - str);
maskstr++;
}
else {
ipstr = str;
}
family = get_netaddr(ipstr, ip);
if (family < 0)
return NULL;
if ((inet_ntop(family, ip, dst, sizeof(dst))) == NULL)
return NULL;
if (maskstr == NULL)
goto out;
/* Parse netmask */
if (parse_int(maskstr, &mask) != 0)
return NULL;
if (mask > max_netmask(family) || mask < 0)
return NULL;
sprintf(dst + strlen(dst), "/%d", mask);
out:
return dst;
}
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(void)
{
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(void)
{
FILE *fd;
char str[128];
int ncpu = 0;
if ((fd = fopen(PROCCPU, "r")) == NULL) {
logger(-1, errno, "Cannot open " PROCCPU);
return 1;
}
while (fgets(str, sizeof(str), fd)) {
char const proc_ptrn[] = "processor";
if (!strncmp(str, proc_ptrn, sizeof(proc_ptrn) - 1))
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;
}
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 get_state_file(unsigned veid, char *buf, int size)
{
return snprintf(buf, size, "%s/%d", VEPIDDIR, 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 to skip (-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);
close(fd);
} else {
close(0); close(1); close(2);
}
}
/* build array of skipped fds */
va_start(ap, close_std);
skip_fds[0] = -1;
for (i = 0; i < ARRAY_SIZE(skip_fds); 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), VPSCONFDIR "/%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;
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 (sscanf(basename(content), "%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;
}
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;
}
const char* ubcstr(unsigned long bar, unsigned long lim)
{
static char str[64];
char *p = str;
char *e = p + sizeof(str) - 1;
#define PRINT_UBC(val) \
if (val == LONG_MAX) \
p += snprintf(p, e - p, "unlimited"); \
else \
p += snprintf(p, e - p, "%lu", val)
PRINT_UBC(bar);
if (bar == lim)
return str;
*p++=':';
PRINT_UBC(lim);
return str;
}
int is_vswap_mode(void)
{
return (access("/proc/vz/vswap", F_OK) == 0);
}
#define UUID_LEN 36
/* Check for valid GUID, add curly brackets if necessary:
* fbcdf284-5345-416b-a589-7b5fcaa87673 ->
* {fbcdf284-5345-416b-a589-7b5fcaa87673}
*/
int vzctl_get_normalized_guid(const char *str, char *buf, int len)
{
int i;
const char *in;
char *out;
if (len < UUID_LEN + 3)
return -1;
in = (str[0] == '{') ? str + 1 : str;
buf[0] = '{';
out = buf + 1;
for (i = 0; i < UUID_LEN; i++) {
if (in[i] == '\0')
break;
if ((i == 8) || (i == 13) || (i == 18) || (i == 23)) {
if (in[i] != '-' )
break;
} else if (!isxdigit(in[i])) {
break;
}
out[i] = in[i];
}
if (i < UUID_LEN)
return 1;
if (in[UUID_LEN] != '\0' &&
(in[UUID_LEN] != '}' || in[UUID_LEN + 1] != '\0'))
return 1;
out[UUID_LEN] = '}';
out[UUID_LEN+1] = '\0';
return 0;
}