blob: d28d9b7b37cbb69e231d4ff8da582d134b8dbb1a [file]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.
*
*/
#include "qpid/log/Statement.h"
#include "qpid/sys/SystemInfo.h"
#include "qpid/sys/posix/check.h"
#include "qpid/sys/posix/PrivatePosix.h"
#include <procinfo.h>
#include <arpa/inet.h>
#include <net/if.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <sys/utsname.h>
#include <unistd.h>
#include <map>
#include <netdb.h>
#include <string.h>
#ifndef HOST_NAME_MAX
# define HOST_NAME_MAX 256
#endif
using namespace std;
namespace qpid {
namespace sys {
long SystemInfo::concurrency() {
return sysconf(_SC_NPROCESSORS_ONLN);
}
bool SystemInfo::getLocalHostname (Address &address) {
char name[HOST_NAME_MAX];
if (::gethostname(name, sizeof(name)) != 0)
return false;
address.host = name;
return true;
}
static const string LOOPBACK("127.0.0.1");
static const string TCP("tcp");
// Test IPv4 address for loopback
inline bool IN_IS_ADDR_LOOPBACK(const ::in_addr* a) {
return ((ntohl(a->s_addr) & 0xff000000) == 0x7f000000);
}
inline bool isLoopback(const ::sockaddr* addr) {
switch (addr->sa_family) {
case AF_INET: return IN_IS_ADDR_LOOPBACK(&((const ::sockaddr_in*)(const void*)addr)->sin_addr);
case AF_INET6: return IN6_IS_ADDR_LOOPBACK(&((const ::sockaddr_in6*)(const void*)addr)->sin6_addr);
default: return false;
}
}
namespace {
class HandleCloser : public IOHandle {
public:
HandleCloser(int fd) : IOHandle(fd) {}
~HandleCloser() { ::close(fd); fd = -1; }
};
inline bool isInetOrInet6(::sockaddr* sa) {
switch (sa->sa_family) {
case AF_INET:
case AF_INET6:
return true;
default:
return false;
}
}
inline void *InetAddr(::sockaddr* sa) {
switch (sa->sa_family) {
case AF_INET:
return &(reinterpret_cast<struct sockaddr_in *>(sa)->sin_addr);
case AF_INET6:
return &(reinterpret_cast<struct sockaddr_in6 *>(sa)->sin6_addr);
default:
return 0;
}
}
typedef std::map<std::string, std::vector<std::string> > InterfaceInfo;
std::map<std::string, std::vector<std::string> > cachedInterfaces;
void cacheInterfaceInfo() {
int status = 0;
int handle = ::socket (PF_INET, SOCK_DGRAM, 0);
QPID_POSIX_CHECK(handle);
HandleCloser h(handle);
size_t num_ifs = 0;
struct ifconf ifc;
status = ::ioctl(handle, SIOCGSIZIFCONF, (caddr_t)&ifc.ifc_len);
QPID_POSIX_CHECK(status);
std::auto_ptr<char> auto_ifc_buf(new char[ifc.ifc_len]);
memset (auto_ifc_buf.get(), 0, ifc.ifc_len);
status = ::ioctl(handle, SIOCGIFCONF, (caddr_t)auto_ifc_buf.get());
QPID_POSIX_CHECK(status);
char *buf_start = auto_ifc_buf.get();
char *buf_end = buf_start + ifc.ifc_len;
for (char *ptr = buf_start; ptr < buf_end; ) {
struct ifreq *req = reinterpret_cast<struct ifreq *>(ptr);
ptr += IFNAMSIZ;
ptr += req->ifr_addr.sa_len;
if (!strcmp("lo0", req->ifr_name) || !isInetOrInet6(&req->ifr_addr))
continue;
char dots[INET6_ADDRSTRLEN];
if (! ::inet_ntop(req->ifr_addr.sa_family, InetAddr(&req->ifr_addr), dots, sizeof(dots)))
throw QPID_POSIX_ERROR(errno);
std::string address(dots);
cachedInterfaces[req->ifr_name].push_back(address);
}
}
}
bool SystemInfo::getInterfaceAddresses(const std::string& interface, std::vector<std::string>& addresses) {
if ( cachedInterfaces.empty() ) cacheInterfaceInfo();
InterfaceInfo::iterator i = cachedInterfaces.find(interface);
if ( i==cachedInterfaces.end() ) return false;
std::copy(i->second.begin(), i->second.end(), std::back_inserter(addresses));
return true;
}
void SystemInfo::getInterfaceNames(std::vector<std::string>& names ) {
if ( cachedInterfaces.empty() ) cacheInterfaceInfo();
for (InterfaceInfo::const_iterator i = cachedInterfaces.begin(); i!=cachedInterfaces.end(); ++i) {
names.push_back(i->first);
}
}
void SystemInfo::getSystemId (std::string &osName,
std::string &nodeName,
std::string &release,
std::string &version,
std::string &machine)
{
struct utsname _uname;
if (uname (&_uname) == 0)
{
osName = _uname.sysname;
nodeName = _uname.nodename;
release = _uname.release;
version = _uname.version;
machine = _uname.machine;
}
}
uint32_t SystemInfo::getProcessId()
{
return (uint32_t) ::getpid();
}
uint32_t SystemInfo::getParentProcessId()
{
return (uint32_t) ::getppid();
}
// AIX specific
string SystemInfo::getProcessName()
{
struct procsinfo my_info;
pid_t my_pid = getpid();
int status = getprocs(&my_info, sizeof(my_info), 0, 0, &my_pid, 1);
QPID_POSIX_CHECK(status);
std::string my_name(my_info.pi_comm);
return my_name;
}
// Always true. Only Windows has exception cases.
bool SystemInfo::threadSafeShutdown()
{
return true;
}
}} // namespace qpid::sys