]> git.saurik.com Git - apple/syslog.git/blobdiff - syslogd.tproj/udp_in.c
syslog-385.tar.gz
[apple/syslog.git] / syslogd.tproj / udp_in.c
index fa91a6968f814c313dbe017169f36a06ca011eb7..4874748b8f66c5348985603389d2caddf7e93383 100644 (file)
@@ -1,27 +1,32 @@
 /*
- * Copyright (c) 2004 Apple Computer, Inc. All rights reserved.
+ * Copyright (c) 2004-2011 Apple Inc. All rights reserved.
  *
  * @APPLE_LICENSE_HEADER_START@
  * 
- * "Portions Copyright (c) 2004 Apple Computer, Inc.  All Rights
- * Reserved.  This file contains Original Code and/or Modifications of
- * Original Code as defined in and that are subject to the Apple Public
- * Source License Version 1.0 (the 'License').  You may not use this file
- * except in compliance with the License.  Please obtain a copy of the
- * License at http://www.apple.com/publicsource and read it before using
- * this file.
+ * This file contains Original Code and/or Modifications of Original Code
+ * as defined in and that are subject to the Apple Public Source License
+ * Version 2.0 (the 'License'). You may not use this file except in
+ * compliance with the License. Please obtain a copy of the License at
+ * http://www.opensource.apple.com/apsl/ and read it before using this
+ * file.
  * 
  * The Original Code and all software distributed under the License are
  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT.  Please see the
- * License for the specific language governing rights and limitations
- * under the License."
+ * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
+ * Please see the License for the specific language governing rights and
+ * limitations under the License.
  * 
  * @APPLE_LICENSE_HEADER_END@
  */
 
+#include <TargetConditionals.h>
+
+#if TARGET_OS_SIMULATOR
+struct _not_empty;
+#else
+
 #include <sys/types.h>
 #include <sys/stat.h>
 #include <sys/socket.h>
 
 #define forever for(;;)
 
+#define UDP_SOCKET_NAME "NetworkListener"
 #define MY_ID "udp_in"
 #define MAXLINE 4096
 
 #define MAXSOCK 16
 static int nsock = 0;
 static int ufd[MAXSOCK];
+static dispatch_source_t ufd_src[MAXSOCK];
 
 static char uline[MAXLINE + 1];
 
+static dispatch_source_t in_src[MAXSOCK];
+static dispatch_queue_t in_queue;
+
 #define FMT_LEGACY 0
 #define FMT_ASL 1
 
-asl_msg_t *
-udp_convert(int fmt, char *s, int len, char *from)
-{
-       char *out;
-       asl_msg_t *m;
-
-       out = NULL;
-       m = NULL;
-
-       if (fmt == FMT_ASL)
-       {
-               m = asl_msg_from_string(s);
-               if (from != NULL) asl_set(m, ASL_KEY_HOST, from);
-               return m;
-       }
-
-       return asl_syslog_input_convert(uline, len, from, 0);
-}
-
-asl_msg_t *
+void 
 udp_in_acceptmsg(int fd)
 {
-       int format, status, x, fromlen;
-       size_t off;
+       socklen_t fromlen;
        ssize_t len;
        struct sockaddr_storage from;
        char fromstr[64], *r, *p;
        struct sockaddr_in *s4;
        struct sockaddr_in6 *s6;
+       asl_msg_t *m;
 
        fromlen = sizeof(struct sockaddr_storage);
        memset(&from, 0, fromlen);
 
        len = recvfrom(fd, uline, MAXLINE, 0, (struct sockaddr *)&from, &fromlen);
-       if (len <= 0) return NULL;
+       if (len <= 0) return;
 
        fromstr[0] = '\0';
        r = NULL;
@@ -96,14 +87,14 @@ udp_in_acceptmsg(int fd)
                s4 = (struct sockaddr_in *)&from;
                inet_ntop(from.ss_family, &(s4->sin_addr), fromstr, 64);
                r = fromstr;
-               asldebug("%s: recvfrom %s len %d\n", MY_ID, fromstr, len);
+               asldebug("%s: fd %d recvfrom %s len %d\n", MY_ID, fd, fromstr, len);
        }
        else if (from.ss_family == AF_INET6)
        {
                s6 = (struct sockaddr_in6 *)&from;
                inet_ntop(from.ss_family, &(s6->sin6_addr), fromstr, 64);
                r = fromstr;
-               asldebug("%s: recvfrom %s len %d\n", MY_ID, fromstr, len);
+               asldebug("%s: fd %d recvfrom %s len %d\n", MY_ID, fd, fromstr, len);
        }
 
        uline[len] = '\0';
@@ -111,83 +102,85 @@ udp_in_acceptmsg(int fd)
        p = strrchr(uline, '\n');
        if (p != NULL) *p = '\0';
 
-
-       /*
-        * Determine if the input is "old" syslog format or new ASL format.
-        * Old format lines should start with "<", but they can just be
-        * straight text.  ASL input starts with a length (10 bytes)
-        * followed by a space and a '['.
-        */
-       format = FMT_LEGACY;
-       off = 0;
-
-       if ((uline[0] != '<') && (len > 11))
-       {
-               status = sscanf(uline, "%d ", &x);
-               if (status == 1) 
-               {
-                       if ((uline[10] == ' ') && (uline[11] == '['))
-                       {
-                               format = FMT_ASL;
-                               off = 11;
-                       }
-               }
-       }
-
-       return udp_convert(format, uline+off, len-off, r);
+       m = asl_input_parse(uline, len, r, SOURCE_UDP_SOCKET);
+       process_message(m, SOURCE_UDP_SOCKET);
 }
 
 int
-udp_in_init(void)
+udp_in_init()
 {
-       struct addrinfo hints, *gai, *ai;
-       int status, i;
+       int i, rbufsize, len, fd;
+       launch_data_t sockets_dict, fd_array, fd_dict;
+       static dispatch_once_t once;
+
+       dispatch_once(&once, ^{
+               in_queue = dispatch_queue_create(MY_ID, NULL);
+       });
 
        asldebug("%s: init\n", MY_ID);
        if (nsock > 0) return 0;
 
-       memset(&hints, 0, sizeof(hints));
-       hints.ai_flags = AI_PASSIVE;
-       hints.ai_family = PF_UNSPEC;
-       hints.ai_socktype = SOCK_DGRAM;
+       if (global.launch_dict == NULL)
+       {
+               asldebug("%s: launchd dict is NULL\n", MY_ID);
+               return -1;
+       }
+
+       sockets_dict = launch_data_dict_lookup(global.launch_dict, LAUNCH_JOBKEY_SOCKETS);
+       if (sockets_dict == NULL)
+       {
+               asldebug("%s: launchd lookup of LAUNCH_JOBKEY_SOCKETS failed\n", MY_ID);
+               return -1;
+       }
+
+       fd_array = launch_data_dict_lookup(sockets_dict, UDP_SOCKET_NAME);
+       if (fd_array == NULL)
+       {
+               asldebug("%s: launchd lookup of UDP_SOCKET_NAME failed\n", MY_ID);
+               return -1;
+       }
 
-       status = getaddrinfo(NULL, "syslog", &hints, &gai);
-       if (status != 0) return -1;
+       nsock = launch_data_array_get_count(fd_array);
+       if (nsock <= 0)
+       {
+               asldebug("%s: launchd fd array is empty\n", MY_ID);
+               return -1;
+       }
 
-       for (ai = gai; (ai != NULL) && (nsock < MAXSOCK); ai = ai->ai_next)
+       for (i = 0; i < nsock; i++)
        {
-               ufd[nsock] = socket(ai->ai_family, ai->ai_socktype, ai->ai_protocol);
-               if (ufd[nsock] < 0)
+               ufd[i] = -1;
+
+               fd_dict = launch_data_array_get_index(fd_array, i);
+               if (fd_dict == NULL)
                {
-                       asldebug("%s: socket: %s\n", MY_ID, strerror(errno));
-                       continue;
+                       asldebug("%s: launchd file discriptor array element 0 is NULL\n", MY_ID);
+                       return -1;
                }
 
-               if (bind(ufd[nsock], ai->ai_addr, ai->ai_addrlen) < 0)
+               fd = launch_data_get_fd(fd_dict);
+
+               rbufsize = 128 * 1024;
+               len = sizeof(rbufsize);
+
+               if (setsockopt(fd, SOL_SOCKET, SO_RCVBUF, &rbufsize, len) < 0)
                {
-                       asldebug("%s: bind: %s\n", MY_ID, strerror(errno));
-                       close(ufd[nsock]);
-                       continue;
+                       asldebug("%s: couldn't set receive buffer size for file descriptor %d: %s\n", MY_ID, fd, strerror(errno));
                }
 
-               nsock++;
-       }
+               if (fcntl(fd, F_SETFL, O_NONBLOCK) < 0)
+               {
+                       asldebug("%s: couldn't set O_NONBLOCK for file descriptor %d: %s\n", MY_ID, fd, strerror(errno));
+               }
 
-       freeaddrinfo(gai);
+               ufd[i] = fd;
 
-       if (nsock == 0)
-       {
-               asldebug("%s: no input sockets\n", MY_ID);
-               return -1;
-       }
+               in_src[i] = dispatch_source_create(DISPATCH_SOURCE_TYPE_READ, (uintptr_t)fd, 0, in_queue);
+               dispatch_source_set_event_handler(in_src[i], ^{ udp_in_acceptmsg(fd); });
 
-       for (i = 0; i < nsock; i++) aslevent_addfd(ufd[i], udp_in_acceptmsg, NULL, NULL);
-       return 0;
-}
+               dispatch_resume(in_src[i]);
+       }
 
-int
-udp_in_reset(void)
-{
        return 0;
 }
 
@@ -196,15 +189,34 @@ udp_in_close(void)
 {
        int i;
 
-       if (nsock == 0) return 1;
+       if (nsock == 0) return -1;
 
        for (i = 0; i < nsock; i++)
        {
-               close(ufd[i]);
-               ufd[i] = -1;
+               if (ufd_src[i] != NULL)
+               {
+                       dispatch_source_cancel(in_src[i]);
+                       dispatch_release(in_src[i]);
+                       in_src[i] = NULL;
+               }
+
+               if (ufd[i] != -1)
+               {
+                       close(ufd[i]);
+                       ufd[i] = -1;
+               }
        }
 
        nsock = 0;
 
        return 0;
 }
+
+int
+udp_in_reset(void)
+{
+       if (udp_in_close() != 0) return -1;
+       return udp_in_init();
+}
+
+#endif /* !TARGET_OS_SIMULATOR */