blob: 90ca52e74ac54d397f534e3c0ca996bd41a359db [file] [log] [blame]
San Mehatfa644ff2009-05-08 11:15:53 -07001#include <alloca.h>
2#include <errno.h>
Kenny Root30abb722010-09-14 14:26:12 -07003#include <sys/socket.h>
San Mehatfa644ff2009-05-08 11:15:53 -07004#include <sys/types.h>
5#include <pthread.h>
San Mehatd7680662009-05-12 11:16:59 -07006#include <string.h>
San Mehatfa644ff2009-05-08 11:15:53 -07007
8#define LOG_TAG "SocketClient"
9#include <cutils/log.h>
10
11#include <sysutils/SocketClient.h>
12
Kenny Root30abb722010-09-14 14:26:12 -070013SocketClient::SocketClient(int socket)
14 : mSocket(socket)
15 , mPid(-1)
16 , mUid(-1)
17 , mGid(-1)
Brad Fitzpatrick13aa8ad2011-03-17 15:41:20 -070018 , mRefCount(1)
Kenny Root30abb722010-09-14 14:26:12 -070019{
San Mehatfa644ff2009-05-08 11:15:53 -070020 pthread_mutex_init(&mWriteMutex, NULL);
Brad Fitzpatrick13aa8ad2011-03-17 15:41:20 -070021 pthread_mutex_init(&mRefCountMutex, NULL);
Kenny Root30abb722010-09-14 14:26:12 -070022
23 struct ucred creds;
24 socklen_t szCreds = sizeof(creds);
25 memset(&creds, 0, szCreds);
26
27 int err = getsockopt(socket, SOL_SOCKET, SO_PEERCRED, &creds, &szCreds);
28 if (err == 0) {
29 mPid = creds.pid;
30 mUid = creds.uid;
31 mGid = creds.gid;
32 }
San Mehatfa644ff2009-05-08 11:15:53 -070033}
34
San Mehatdb017542009-05-20 15:27:14 -070035int SocketClient::sendMsg(int code, const char *msg, bool addErrno) {
San Mehatd7680662009-05-12 11:16:59 -070036 char *buf;
David 'Digit' Turner01feae22011-01-17 02:34:15 +010037 const char* arg;
38 const char* fmt;
39 char tmp[1];
40 int len;
San Mehat03f0d272009-05-26 15:18:25 -070041
San Mehatd7680662009-05-12 11:16:59 -070042 if (addErrno) {
David 'Digit' Turner01feae22011-01-17 02:34:15 +010043 fmt = "%.3d %s (%s)";
44 arg = strerror(errno);
San Mehatd7680662009-05-12 11:16:59 -070045 } else {
David 'Digit' Turner01feae22011-01-17 02:34:15 +010046 fmt = "%.3d %s";
47 arg = NULL;
San Mehatd7680662009-05-12 11:16:59 -070048 }
David 'Digit' Turner01feae22011-01-17 02:34:15 +010049 /* Measure length of required buffer */
50 len = snprintf(tmp, sizeof tmp, fmt, code, msg, arg);
51 /* Allocate in the stack, then write to it */
52 buf = (char*)alloca(len+1);
53 snprintf(buf, len+1, fmt, code, msg, arg);
54 /* Send the zero-terminated message */
San Mehatd7680662009-05-12 11:16:59 -070055 return sendMsg(buf);
56}
57
San Mehatdb017542009-05-20 15:27:14 -070058int SocketClient::sendMsg(const char *msg) {
San Mehatfa644ff2009-05-08 11:15:53 -070059 if (mSocket < 0) {
60 errno = EHOSTUNREACH;
61 return -1;
62 }
63
San Mehatc73a3a52009-06-15 14:06:03 -070064 // Send the message including null character
Brad Fitzpatricka2a1f312010-10-27 10:23:16 -070065 if (sendData(msg, strlen(msg) + 1) != 0) {
66 SLOGW("Unable to send msg '%s'", msg);
67 return -1;
68 }
69 return 0;
70}
71
72int SocketClient::sendData(const void* data, int len) {
San Mehatd7680662009-05-12 11:16:59 -070073 int rc = 0;
Brad Fitzpatricka2a1f312010-10-27 10:23:16 -070074 const char *p = (const char*) data;
75 int brtw = len;
San Mehatd7680662009-05-12 11:16:59 -070076
Brad Fitzpatrickb9634d02010-11-02 10:55:52 -070077 if (len == 0) {
78 return 0;
79 }
80
San Mehatfa644ff2009-05-08 11:15:53 -070081 pthread_mutex_lock(&mWriteMutex);
Brad Fitzpatricka2a1f312010-10-27 10:23:16 -070082 while (brtw > 0) {
David 'Digit' Turner01feae22011-01-17 02:34:15 +010083 rc = write(mSocket, p, brtw);
84 if (rc > 0) {
85 p += rc;
86 brtw -= rc;
87 continue;
88 }
89
90 if (rc < 0 && errno == EINTR)
91 continue;
92
93 pthread_mutex_unlock(&mWriteMutex);
94 if (rc == 0) {
San Mehat7e8529a2010-03-25 09:31:42 -070095 SLOGW("0 length write :(");
San Mehatd7680662009-05-12 11:16:59 -070096 errno = EIO;
David 'Digit' Turner01feae22011-01-17 02:34:15 +010097 } else {
98 SLOGW("write error (%s)", strerror(errno));
San Mehatd7680662009-05-12 11:16:59 -070099 }
David 'Digit' Turner01feae22011-01-17 02:34:15 +0100100 return -1;
San Mehatfa644ff2009-05-08 11:15:53 -0700101 }
102 pthread_mutex_unlock(&mWriteMutex);
103 return 0;
104}
Brad Fitzpatrick13aa8ad2011-03-17 15:41:20 -0700105
106void SocketClient::incRef() {
Brad Fitzpatrick3549b0d2011-03-17 17:14:46 -0700107 pthread_mutex_lock(&mRefCountMutex);
108 mRefCount++;
109 pthread_mutex_unlock(&mRefCountMutex);
Brad Fitzpatrick13aa8ad2011-03-17 15:41:20 -0700110}
111
Brad Fitzpatrick3549b0d2011-03-17 17:14:46 -0700112bool SocketClient::decRef() {
113 bool deleteSelf = false;
114 pthread_mutex_lock(&mRefCountMutex);
115 mRefCount--;
116 if (mRefCount == 0) {
117 deleteSelf = true;
118 } else if (mRefCount < 0) {
119 SLOGE("SocketClient refcount went negative!");
120 }
121 pthread_mutex_unlock(&mRefCountMutex);
122 if (deleteSelf) {
123 delete this;
124 }
125 return deleteSelf;
Brad Fitzpatrick13aa8ad2011-03-17 15:41:20 -0700126}