Arquivos
hhvm/hphp/runtime/ext/ext_posix.cpp
T
smith c646a30002 Litstr must die, episode III.
Many callsites of Array.set(litstr, ...)
2013-04-23 12:57:40 -07:00

428 linhas
11 KiB
C++

/*
+----------------------------------------------------------------------+
| HipHop for PHP |
+----------------------------------------------------------------------+
| Copyright (c) 2010- Facebook, Inc. (http://www.facebook.com) |
| Copyright (c) 1997-2010 The PHP Group |
+----------------------------------------------------------------------+
| This source file is subject to version 3.01 of the PHP license, |
| that is bundled with this package in the file LICENSE, and is |
| available through the world-wide-web at the following url: |
| http://www.php.net/license/3_01.txt |
| If you did not receive a copy of the PHP license and are unable to |
| obtain it through the world-wide-web, please send a note to |
| license@php.net so we can mail you a copy immediately. |
+----------------------------------------------------------------------+
*/
#include <runtime/ext/ext_posix.h>
#include <runtime/base/file/file.h>
#include <sys/times.h>
#include <sys/utsname.h>
#include <sys/types.h>
#include <sys/time.h>
#include <pwd.h>
#include <memory>
namespace HPHP {
IMPLEMENT_DEFAULT_EXTENSION(posix);
///////////////////////////////////////////////////////////////////////////////
bool f_posix_access(CStrRef file, int mode /* = 0 */) {
String path = File::TranslatePath(file);
if (path.empty()) {
return false;
}
return !access(path.data(), mode);
}
String f_posix_ctermid() {
String s = String(L_ctermid, ReserveString);
char *buffer = s.mutableSlice().ptr;
ctermid(buffer);
return s.setSize(strlen(buffer));
}
int64_t f_posix_get_last_error() {
return errno;
}
String f_posix_getcwd() {
String s = String(PATH_MAX, ReserveString);
char *buffer = s.mutableSlice().ptr;
if (getcwd(buffer, PATH_MAX) == NULL) {
return "/";
}
return s.setSize(strlen(buffer));
}
int64_t f_posix_getegid() {
return getegid();
}
int64_t f_posix_geteuid() {
return geteuid();
}
int64_t f_posix_getgid() {
return getgid();
}
static const StaticString s_name("name");
static const StaticString s_passwd("passwd");
static const StaticString s_members("members");
static const StaticString s_uid("uid");
static const StaticString s_gid("gid");
static const StaticString s_gecos("gecos");
static const StaticString s_dir("dir");
static const StaticString s_shell("shell");
static Variant php_posix_group_to_array(int gid,
CStrRef gname = null_variant) {
// Don't pass a gid *and* a gname to this.
assert((gid < 0) || gname.size() == 0);
if ((gid < 0) && (gname.size() == 0)) {
return false;
}
int grbuflen = sysconf(_SC_GETGR_R_SIZE_MAX);
if (grbuflen < 1) {
return false;
}
std::unique_ptr<char[]> grbuf(new char[grbuflen]);
struct group gr;
struct group *retgrptr = NULL;
// If we somehow reach this point and both gname and gid were
// passed, then the gid values will override the game values,
// but it will otherwise function just fine.
// The assert() clause above should prevent that, however.
if ((gname.size() > 0) &&
getgrnam_r(gname.data(), &gr, grbuf.get(), grbuflen, &retgrptr)) {
return false;
} else if ((gid >= 0) &&
getgrgid_r(gid, &gr, grbuf.get(), grbuflen, &retgrptr)) {
return false;
}
Array members;
for (int count=0; gr.gr_mem[count] != NULL; count++) {
members.append(String(gr.gr_mem[count], CopyString));
}
ArrayInit ret(4);
ret.set(s_name, String(gr.gr_name, CopyString));
ret.set(s_passwd, String(gr.gr_passwd, CopyString));
ret.set(s_members, members);
ret.set(s_gid, (int)gr.gr_gid);
return ret.create();
}
Variant f_posix_getgrgid(int gid) {
return php_posix_group_to_array(gid);
}
Variant f_posix_getgrnam(CStrRef name) {
return php_posix_group_to_array(-1, name.data());
}
Variant f_posix_getgroups() {
gid_t gidlist[NGROUPS_MAX];
int result = getgroups(NGROUPS_MAX, gidlist);
if (result < 0) {
return false;
}
Array ret;
for (int i = 0; i < result; i++) {
ret.append((int)gidlist[i]);
}
return ret;
}
Variant f_posix_getlogin() {
char buf[L_cuserid];
if (!getlogin_r(buf, sizeof(buf) - 1)) {
return String(buf, CopyString);
}
return false;
}
Variant f_posix_getpgid(int pid) {
int ret = getpgid(pid);
if (ret < 0) return false;
return ret;
}
int64_t f_posix_getpgrp() {
return getpgrp();
}
int64_t f_posix_getpid() {
return getpid();
}
int64_t f_posix_getppid() {
return getppid();
}
static Variant php_posix_passwd_to_array(int uid,
CStrRef name = null_variant) {
// Don't pass a uid *and* a name to this.
assert((uid < 0) || name.size() == 0);
if ((uid < 0) && name.size() == 0) {
return false;
}
int pwbuflen = sysconf(_SC_GETPW_R_SIZE_MAX);
if (pwbuflen < 1) {
return false;
}
std::unique_ptr<char[]> pwbuf(new char[pwbuflen]);
struct passwd pw;
struct passwd *retpwptr = NULL;
// If we somehow reach this point and both name and uid were
// passed, then the uid values will override the name values,
// but it will otherwise function just fine.
// The assert() clauses above should prevent that, however.
if ((name.size() > 0) &&
getpwnam_r(name.data(), &pw, pwbuf.get(), pwbuflen, &retpwptr)) {
return false;
} else if ((uid >= 0) &&
getpwuid_r(uid, &pw, pwbuf.get(), pwbuflen, &retpwptr)) {
return false;
}
ArrayInit ret(7);
ret.set(s_name, String(pw.pw_name, CopyString));
ret.set(s_passwd, String(pw.pw_passwd, CopyString));
ret.set(s_uid, (int)pw.pw_uid);
ret.set(s_gid, (int)pw.pw_gid);
ret.set(s_gecos, String(pw.pw_gecos, CopyString));
ret.set(s_dir, String(pw.pw_dir, CopyString));
ret.set(s_shell, String(pw.pw_shell, CopyString));
return ret.create();
}
Variant f_posix_getpwnam(CStrRef username) {
return php_posix_passwd_to_array(-1, username);
}
Variant f_posix_getpwuid(int uid) {
return php_posix_passwd_to_array(uid);
}
static bool posix_addlimit(int limit, const char *name, Array &ret) {
char hard[80]; snprintf(hard, 80, "hard %s", name);
char soft[80]; snprintf(soft, 80, "soft %s", name);
struct rlimit rl;
int result = getrlimit(limit, &rl);
if (result < 0) {
return false;
}
String softStr(soft, CopyString);
String hardStr(hard, CopyString);
if (rl.rlim_cur == RLIM_INFINITY) {
ret.set(softStr, "unlimited");
} else {
ret.set(softStr, (int)rl.rlim_cur);
}
if (rl.rlim_max == RLIM_INFINITY) {
ret.set(hardStr, "unlimited");
} else {
ret.set(hardStr, (int)rl.rlim_max);
}
return true;
}
static struct limitlist {
int limit;
const char *name;
} limits[] = {
{ RLIMIT_CORE, "core" },
{ RLIMIT_DATA, "data" },
{ RLIMIT_STACK, "stack" },
//{ RLIMIT_VMEM, "virtualmem" },
{ RLIMIT_AS, "totalmem" },
{ RLIMIT_RSS, "rss" },
{ RLIMIT_NPROC, "maxproc" },
{ RLIMIT_MEMLOCK, "memlock" },
{ RLIMIT_CPU, "cpu" },
{ RLIMIT_FSIZE, "filesize" },
{ RLIMIT_NOFILE, "openfiles" },
{ 0, NULL }
};
Variant f_posix_getrlimit() {
Array ret;
for (struct limitlist *l = limits; l->name; l++) {
if (!posix_addlimit(l->limit, l->name, ret)) {
return false;
}
}
return ret;
}
Variant f_posix_getsid(int pid) {
int ret = getsid(pid);
if (ret < 0) return false;
return ret;
}
int64_t f_posix_getuid() {
return getuid();
}
bool f_posix_initgroups(CStrRef name, int base_group_id) {
if (name.empty()) return false;
return !initgroups(name.data(), base_group_id);
}
static int php_posix_get_fd(CVarRef fd) {
int nfd;
if (fd.isResource()) {
File *f = fd.toObject().getTyped<File>();
if (!f) {
return false;
}
nfd = f->fd();
} else {
nfd = fd.toInt32();
}
return nfd;
}
bool f_posix_isatty(CVarRef fd) {
return isatty(php_posix_get_fd(fd));
}
bool f_posix_kill(int pid, int sig) {
return kill(pid, sig) >= 0;
}
bool f_posix_mkfifo(CStrRef pathname, int mode) {
return mkfifo(pathname.data(), mode) >= 0;
}
bool f_posix_mknod(CStrRef pathname, int mode, int major /* = 0 */,
int minor /* = 0 */) {
dev_t php_dev = 0;
if ((mode & S_IFCHR) || (mode & S_IFBLK)) {
if (major == 0 && minor == 0) {
raise_warning("For S_IFCHR and S_IFBLK you need to pass "
"a major device kernel identifier");
return false;
}
if (major == 0) {
raise_warning("Expects argument 3 to be non-zero for "
"POSIX_S_IFCHR and POSIX_S_IFBLK");
return false;
}
php_dev = makedev(major, minor);
}
return mknod(pathname.data(), mode, php_dev) >= 0;
}
bool f_posix_setegid(int gid) {
return setegid(gid);
}
bool f_posix_seteuid(int uid) {
return seteuid(uid);
}
bool f_posix_setgid(int gid) {
return setgid(gid);
}
bool f_posix_setpgid(int pid, int pgid) {
return setpgid(pid, pgid) >= 0;
}
int64_t f_posix_setsid() {
return setsid();
}
bool f_posix_setuid(int uid) {
return setuid(uid);
}
String f_posix_strerror(int errnum) {
return String(Util::safe_strerror(errnum));
}
static const StaticString s_ticks("ticks");
static const StaticString s_utime("utime");
static const StaticString s_stime("stime");
static const StaticString s_cutime("cutime");
static const StaticString s_cstime("cstime");
Variant f_posix_times() {
struct tms t;
clock_t ticks = times(&t);
if (ticks == -1) {
return false;
}
ArrayInit ret(5);
ret.set(s_ticks, (int)ticks); /* clock ticks */
ret.set(s_utime, (int)t.tms_utime); /* user time */
ret.set(s_stime, (int)t.tms_stime); /* system time */
ret.set(s_cutime, (int)t.tms_cutime); /* user time of children */
ret.set(s_cstime, (int)t.tms_cstime); /* system time of children */
return ret.create();
}
Variant f_posix_ttyname(CVarRef fd) {
int ttyname_maxlen = sysconf(_SC_TTY_NAME_MAX);
if (ttyname_maxlen <= 0) {
return false;
}
String ttyname(ttyname_maxlen, ReserveString);
char *p = ttyname.mutableSlice().ptr;
if (ttyname_r(php_posix_get_fd(fd), p, ttyname_maxlen)) {
return false;
}
return ttyname.setSize(strlen(p));
}
static const StaticString s_sysname("sysname");
static const StaticString s_nodename("nodename");
static const StaticString s_release("release");
static const StaticString s_version("version");
static const StaticString s_machine("machine");
static const StaticString s_domainname("domainname");
Variant f_posix_uname() {
struct utsname u;
if (uname(&u) < 0) {
return false;
}
Array ret;
ret.set(s_sysname, String(u.sysname, CopyString));
ret.set(s_nodename, String(u.nodename, CopyString));
ret.set(s_release, String(u.release, CopyString));
ret.set(s_version, String(u.version, CopyString));
ret.set(s_machine, String(u.machine, CopyString));
#if defined(_GNU_SOURCE) && !defined(__APPLE__) && !defined(__FreeBSD__)
ret.set(s_domainname, String(u.domainname, CopyString));
#endif
return ret;
}
///////////////////////////////////////////////////////////////////////////////
}