• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2012-2014 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include <limits.h>
18 #include <sys/cdefs.h>
19 #include <sys/prctl.h>
20 #include <sys/socket.h>
21 #include <sys/types.h>
22 #include <sys/un.h>
23 #include <unistd.h>
24 
25 #include <cutils/sockets.h>
26 #include <log/logger.h>
27 #include <private/android_filesystem_config.h>
28 #include <private/android_logger.h>
29 
30 #include "LogBuffer.h"
31 #include "LogListener.h"
32 #include "LogUtils.h"
33 
LogListener(LogBuffer * buf,LogReader * reader)34 LogListener::LogListener(LogBuffer *buf, LogReader *reader) :
35         SocketListener(getLogSocket(), false),
36         logbuf(buf),
37         reader(reader) {
38 }
39 
onDataAvailable(SocketClient * cli)40 bool LogListener::onDataAvailable(SocketClient *cli) {
41     static bool name_set;
42     if (!name_set) {
43         prctl(PR_SET_NAME, "logd.writer");
44         name_set = true;
45     }
46 
47     char buffer[sizeof_log_id_t + sizeof(uint16_t) + sizeof(log_time)
48         + LOGGER_ENTRY_MAX_PAYLOAD];
49     struct iovec iov = { buffer, sizeof(buffer) };
50 
51     char control[CMSG_SPACE(sizeof(struct ucred))] __aligned(4);
52     struct msghdr hdr = {
53         NULL,
54         0,
55         &iov,
56         1,
57         control,
58         sizeof(control),
59         0,
60     };
61 
62     int socket = cli->getSocket();
63 
64     // To clear the entire buffer is secure/safe, but this contributes to 1.68%
65     // overhead under logging load. We are safe because we check counts.
66     // memset(buffer, 0, sizeof(buffer));
67     ssize_t n = recvmsg(socket, &hdr, 0);
68     if (n <= (ssize_t)(sizeof(android_log_header_t))) {
69         return false;
70     }
71 
72     struct ucred *cred = NULL;
73 
74     struct cmsghdr *cmsg = CMSG_FIRSTHDR(&hdr);
75     while (cmsg != NULL) {
76         if (cmsg->cmsg_level == SOL_SOCKET
77                 && cmsg->cmsg_type  == SCM_CREDENTIALS) {
78             cred = (struct ucred *)CMSG_DATA(cmsg);
79             break;
80         }
81         cmsg = CMSG_NXTHDR(&hdr, cmsg);
82     }
83 
84     if (cred == NULL) {
85         return false;
86     }
87 
88     if (cred->uid == AID_LOGD) {
89         // ignore log messages we send to ourself.
90         // Such log messages are often generated by libraries we depend on
91         // which use standard Android logging.
92         return false;
93     }
94 
95     android_log_header_t *header = reinterpret_cast<android_log_header_t *>(buffer);
96     if (/* header->id < LOG_ID_MIN || */ header->id >= LOG_ID_MAX || header->id == LOG_ID_KERNEL) {
97         return false;
98     }
99 
100     if ((header->id == LOG_ID_SECURITY) &&
101             (!__android_log_security() ||
102              !clientHasLogCredentials(cred->uid, cred->gid, cred->pid))) {
103         return false;
104     }
105 
106     char *msg = ((char *)buffer) + sizeof(android_log_header_t);
107     n -= sizeof(android_log_header_t);
108 
109     // NB: hdr.msg_flags & MSG_TRUNC is not tested, silently passing a
110     // truncated message to the logs.
111 
112     if (logbuf->log((log_id_t)header->id, header->realtime,
113             cred->uid, cred->pid, header->tid, msg,
114             ((size_t) n <= USHRT_MAX) ? (unsigned short) n : USHRT_MAX) >= 0) {
115         reader->notifyNewLog();
116     }
117 
118     return true;
119 }
120 
getLogSocket()121 int LogListener::getLogSocket() {
122     static const char socketName[] = "logdw";
123     int sock = android_get_control_socket(socketName);
124 
125     if (sock < 0) {
126         sock = socket_local_server(socketName,
127                                    ANDROID_SOCKET_NAMESPACE_RESERVED,
128                                    SOCK_DGRAM);
129     }
130 
131     int on = 1;
132     if (setsockopt(sock, SOL_SOCKET, SO_PASSCRED, &on, sizeof(on)) < 0) {
133         return -1;
134     }
135     return sock;
136 }
137