xref: /titanic_41/usr/src/uts/common/fs/sockfs/nl7c.c (revision af33352a19f03e8fe054fe9b44b19c20aeed0da6)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 /*
27  * NL7C (Network Layer 7 Cache) as part of SOCKFS provides an in-kernel
28  * gateway cache for the request/response message based L7 protocol HTTP
29  * (Hypertext Transfer Protocol, see HTTP/1.1 RFC2616) in a semantically
30  * transparent manner.
31  *
32  * Neither the requesting user agent (client, e.g. web browser) nor the
33  * origin server (e.g. webserver) that provided the response cached by
34  * NL7C are impacted in any way.
35  *
36  * Note, currently NL7C only processes HTTP messages via the embedded
37  * URI of scheme http (not https nor any other), additional scheme are
38  * intended to be supported as is practical such that much of the NL7C
39  * framework may appear more general purpose then would be needed just
40  * for an HTTP gateway cache.
41  *
42  * NL7C replaces NCA (Network Cache and Accelerator) and in the future
43  * NCAS (NCA/SSL).
44  *
45  * Further, NL7C uses all NCA configuration files, see "/etc/nca/", the
46  * NCA socket API, "AF_NCA", and "ndd /dev/nca" for backwards compatibility.
47  */
48 
49 #include <sys/systm.h>
50 #include <sys/strsun.h>
51 #include <sys/strsubr.h>
52 #include <inet/common.h>
53 #include <inet/led.h>
54 #include <inet/mi.h>
55 #include <netinet/in.h>
56 #include <fs/sockfs/nl7c.h>
57 #include <fs/sockfs/nl7curi.h>
58 #include <fs/sockfs/socktpi.h>
59 
60 #include <inet/nca/ncadoorhdr.h>
61 #include <inet/nca/ncalogd.h>
62 #include <inet/nca/ncandd.h>
63 
64 #include <sys/promif.h>
65 
66 /*
67  * NL7C, NCA, NL7C logger enabled:
68  */
69 
70 boolean_t	nl7c_enabled = B_FALSE;
71 
72 boolean_t	nl7c_logd_enabled = B_FALSE;
73 boolean_t	nl7c_logd_started = B_FALSE;
74 boolean_t	nl7c_logd_cycle = B_TRUE;
75 
76 /*
77  * Some externs:
78  */
79 
80 extern int	inet_pton(int, char *, void *);
81 
82 extern void	nl7c_uri_init(void);
83 extern boolean_t nl7c_logd_init(int, caddr_t *);
84 extern void	nl7c_nca_init(void);
85 
86 /*
87  * nl7c_addr_t - a singly linked grounded list, pointed to by *nl7caddrs,
88  * constructed at init time by parsing "/etc/nca/ncaport.conf".
89  *
90  * This list is searched at bind(3SOCKET) time when an application doesn't
91  * explicitly set AF_NCA but instead uses AF_INET, if a match is found then
92  * the underlying socket is marked sti_nl7c_flags NL7C_ENABLED.
93  */
94 
95 typedef struct nl7c_addr_s {
96 	struct nl7c_addr_s *next;	/* next entry */
97 	sa_family_t	family;		/* addr type, only INET and INET6 */
98 	uint16_t	port;		/* port */
99 	union {
100 		ipaddr_t	v4;	/* IPv4 address */
101 		in6_addr_t	v6;	/* IPv6 address */
102 		void		*align;	/* foce alignment */
103 	}		addr;		/* address */
104 
105 	struct sonode	*listener;	/* listen()er's sonode */
106 	boolean_t	temp;		/* temporary addr via add_addr() ? */
107 } nl7c_addr_t;
108 
109 nl7c_addr_t	*nl7caddrs = NULL;
110 
111 /*
112  * Called for an NL7C_ENABLED listen()er socket for the nl7c_addr_t
113  * previously returned by nl7c_lookup_addr().
114  */
115 
116 void
117 nl7c_listener_addr(void *arg, struct sonode *so)
118 {
119 	nl7c_addr_t		*p = (nl7c_addr_t *)arg;
120 
121 	if (p->listener == NULL)
122 		p->listener = so;
123 	SOTOTPI(so)->sti_nl7c_addr = arg;
124 }
125 
126 struct sonode *
127 nl7c_addr2portso(void *arg)
128 {
129 	nl7c_addr_t		*p = (nl7c_addr_t *)arg;
130 
131 	return (p->listener);
132 }
133 
134 void *
135 nl7c_lookup_addr(void *addr, t_uscalar_t addrlen)
136 {
137 	struct sockaddr		*sap = addr;
138 	struct sockaddr_in	*v4p = addr;
139 	nl7c_addr_t		*p = nl7caddrs;
140 
141 	if (sap->sa_family != AF_INET || addrlen != sizeof (*v4p)) {
142 		/* Only support IPv4 */
143 		return (B_FALSE);
144 	}
145 	while (p) {
146 		if (sap->sa_family == p->family &&
147 		    v4p->sin_port == p->port &&
148 		    (v4p->sin_addr.s_addr == p->addr.v4 ||
149 		    p->addr.v4 == INADDR_ANY)) {
150 			/* Match */
151 			return (p);
152 		}
153 		p = p->next;
154 	}
155 	return (NULL);
156 }
157 
158 void *
159 nl7c_add_addr(void *addr, t_uscalar_t addrlen)
160 {
161 	struct sockaddr		*sap = addr;
162 	struct sockaddr_in	*v4p = addr;
163 	nl7c_addr_t		*new = NULL;
164 	nl7c_addr_t		*old;
165 	nl7c_addr_t		*p;
166 	boolean_t		alloced;
167 
168 	if (sap->sa_family != AF_INET || addrlen != sizeof (*v4p)) {
169 		/* Only support IPv4 */
170 		return (NULL);
171 	}
172 again:
173 	p = nl7caddrs;
174 	while (p) {
175 		if (new == NULL && p->port == 0)
176 			new = p;
177 		if (sap->sa_family == p->family &&
178 		    v4p->sin_port == p->port &&
179 		    (v4p->sin_addr.s_addr == p->addr.v4 ||
180 		    p->addr.v4 == INADDR_ANY)) {
181 			/* Match */
182 			return (p);
183 		}
184 		p = p->next;
185 	}
186 	if (new == NULL) {
187 		new = kmem_zalloc(sizeof (*new), KM_SLEEP);
188 		alloced = B_TRUE;
189 	} else
190 		alloced = B_FALSE;
191 
192 	new->family = sap->sa_family;
193 	new->port = v4p->sin_port;
194 	new->addr.v4 = v4p->sin_addr.s_addr;
195 	new->temp = B_TRUE;
196 
197 	if (alloced) {
198 		old = nl7caddrs;
199 		new->next = old;
200 		if (atomic_cas_ptr(&nl7caddrs, old, new) != old) {
201 			kmem_free(new, sizeof (*new));
202 			goto again;
203 		}
204 	}
205 
206 	return (new);
207 }
208 
209 boolean_t
210 nl7c_close_addr(struct sonode *so)
211 {
212 	nl7c_addr_t	*p = nl7caddrs;
213 
214 	while (p) {
215 		if (p->listener == so) {
216 			if (p->temp)
217 				p->port = (uint16_t)-1;
218 			p->listener = NULL;
219 			return (B_TRUE);
220 		}
221 		p = p->next;
222 	}
223 	return (B_FALSE);
224 }
225 
226 static void
227 nl7c_addr_add(nl7c_addr_t *p)
228 {
229 	p->next = nl7caddrs;
230 	nl7caddrs = p;
231 }
232 
233 void
234 nl7c_mi_report_addr(mblk_t *mp)
235 {
236 	ipaddr_t	ip;
237 	uint16_t	port;
238 	nl7c_addr_t	*p = nl7caddrs;
239 	struct sonode	*so;
240 	char		addr[32];
241 
242 	(void) mi_mpprintf(mp, "Door  Up-Call-Queue IPaddr:TCPport Listenning");
243 	while (p) {
244 		if (p->port != (uint16_t)-1) {
245 			/* Don't report freed slots */
246 			ip = ntohl(p->addr.v4);
247 			port = ntohs(p->port);
248 
249 			if (ip == INADDR_ANY) {
250 				(void) strcpy(addr, "*");
251 			} else {
252 				int a1 = (ip >> 24) & 0xFF;
253 				int a2 = (ip >> 16) & 0xFF;
254 				int a3 = (ip >> 8) & 0xFF;
255 				int a4 = ip & 0xFF;
256 
257 				(void) mi_sprintf(addr, "%d.%d.%d.%d",
258 				    a1, a2, a3, a4);
259 			}
260 			so = p->listener;
261 			(void) mi_mpprintf(mp, "%p  %s:%d  %d",
262 			    so ? (void *)strvp2wq(SOTOV(so)) : NULL,
263 			    addr, port, p->listener ? 1 : 0);
264 		}
265 		p = p->next;
266 	}
267 }
268 
269 /*
270  * ASCII to unsigned.
271  *
272  * Note, it's assumed that *p is a valid zero byte terminated string.
273  */
274 
275 static unsigned
276 atou(const char *p)
277 {
278 	int c;
279 	int v = 0;
280 
281 	/* Shift and add digit by digit */
282 	while ((c = *p++) != NULL && isdigit(c)) {
283 		v *= 10;
284 		v += c - '0';
285 	}
286 	return (v);
287 }
288 
289 /*
290  * strdup(), yet another strdup() in the kernel.
291  */
292 
293 static char *
294 strdup(char *s)
295 {
296 	int	len = strlen(s) + 1;
297 	char	*ret = kmem_alloc(len, KM_SLEEP);
298 
299 	bcopy(s, ret, len);
300 
301 	return (ret);
302 }
303 
304 /*
305  * Inet ASCII to binary.
306  *
307  * Note, it's assumed that *s is a valid zero byte terminated string, and
308  * that *p is a zero initialized struct (this is important as the value of
309  * INADDR_ANY and IN6ADDR_ANY is zero).
310  */
311 
312 static int
313 inet_atob(char *s, nl7c_addr_t *p)
314 {
315 	if (strcmp(s, "*") == 0) {
316 		/* INADDR_ANY */
317 		p->family = AF_INET;
318 		return (0);
319 	}
320 	if (strcmp(s, "::") == 0) {
321 		/* IN6ADDR_ANY */
322 		p->family = AF_INET6;
323 		return (0);
324 	}
325 	/* IPv4 address ? */
326 	if (inet_pton(AF_INET, s, &p->addr.v4) != 1) {
327 		/* Nop, IPv6 address ? */
328 		if (inet_pton(AF_INET6, s, &p->addr.v6) != 1) {
329 			/* Nop, return error */
330 			return (1);
331 		}
332 		p->family = AF_INET6;
333 	} else {
334 		p->family = AF_INET;
335 		p->addr.v4 = ntohl(p->addr.v4);
336 	}
337 	return (0);
338 }
339 
340 /*
341  * Open and read each line from "/etc/nca/ncaport.conf", the syntax of a
342  * ncaport.conf file line is:
343  *
344  *	ncaport=IPaddr/Port[/Proxy]
345  *
346  * Where:
347  *
348  * ncaport - the only token recognized.
349  *
350  *  IPaddr - an IPv4 numeric dot address (e.g. 192.168.84.71) or '*' for
351  *           INADDR_ANY, or an IPv6 numeric address or "::" for IN6ADDR_ANY.
352  *
353  *       / - IPaddr/Port separator.
354  *
355  *    Port - a TCP decimal port number.
356  *
357  * Note, all other lines will be ignored.
358  */
359 
360 static void
361 ncaportconf_read(void)
362 {
363 	int	ret;
364 	struct vnode *vp;
365 	char	c;
366 	ssize_t resid;
367 	char	buf[1024];
368 	char	*ebp = &buf[sizeof (buf)];
369 	char	*bp = ebp;
370 	offset_t off = 0;
371 	enum parse_e {START, TOK, ADDR, PORT, EOL} parse = START;
372 	nl7c_addr_t *addrp = NULL;
373 	char	*ncaport = "ncaport";
374 	char	string[] = "XXXX:XXXX:XXXX:XXXX:XXXX:XXXX:XXXX:XXXX";
375 	char	*stringp;
376 	char	*tok;
377 	char	*portconf = "/etc/nca/ncaport.conf";
378 
379 	ret = vn_open(portconf, UIO_SYSSPACE, FREAD, 0, &vp, 0, 0);
380 	if (ret == ENOENT) {
381 		/* No portconf file, nothing to do */
382 		return;
383 	}
384 	if (ret != 0) {
385 		/* Error of some sort, tell'm about it */
386 		cmn_err(CE_WARN, "%s: open error %d", portconf, ret);
387 		return;
388 	}
389 	/*
390 	 * Read portconf one buf[] at a time, parse one char at a time.
391 	 */
392 	for (;;) {
393 		if (bp == ebp) {
394 			/* Nothing left in buf[], read another */
395 			ret = vn_rdwr(UIO_READ, vp, buf, sizeof (buf), off,
396 			    UIO_SYSSPACE, 0, (rlim64_t)0, CRED(), &resid);
397 			if (ret != 0) {
398 				/* Error of some sort, tell'm about it */
399 				cmn_err(CE_WARN, "%s: read error %d",
400 				    portconf, ret);
401 				break;
402 			}
403 			if (resid == sizeof (buf)) {
404 				/* EOF, done */
405 				break;
406 			}
407 			/* Initilize per buf[] state */
408 			bp = buf;
409 			ebp = &buf[sizeof (buf) - resid];
410 			off += sizeof (buf) - resid;
411 		}
412 		c = *bp++;
413 		switch (parse) {
414 		case START:
415 			/* Initilize all per file line state */
416 			if (addrp == NULL) {
417 				addrp = kmem_zalloc(sizeof (*addrp),
418 				    KM_NOSLEEP);
419 			}
420 			tok = ncaport;
421 			stringp = string;
422 			parse = TOK;
423 			/*FALLTHROUGH*/
424 		case TOK:
425 			if (c == '#') {
426 				/* Comment through end of line */
427 				parse = EOL;
428 				break;
429 			}
430 			if (isalpha(c)) {
431 				if (c != *tok++) {
432 					/* Only know one token, skip */
433 					parse = EOL;
434 				}
435 			} else if (c == '=') {
436 				if (*tok != NULL) {
437 					/* Only know one token, skip */
438 					parse = EOL;
439 					break;
440 				}
441 				parse = ADDR;
442 			} else if (c == '\n') {
443 				/* Found EOL, empty line, next line */
444 				parse = START;
445 			} else {
446 				/* Unexpected char, skip */
447 				parse = EOL;
448 			}
449 			break;
450 
451 		case ADDR:
452 			if (c == '/') {
453 				/* addr/port separator, end of addr */
454 				*stringp = NULL;
455 				if (inet_atob(string, addrp)) {
456 					/* Bad addr, skip */
457 					parse = EOL;
458 				} else {
459 					stringp = string;
460 					parse = PORT;
461 				}
462 			} else {
463 				/* Save char to string */
464 				if (stringp ==
465 				    &string[sizeof (string) - 1]) {
466 					/* Would overflow, skip */
467 					parse = EOL;
468 				} else {
469 					/* Copy IP addr char */
470 					*stringp++ = c;
471 				}
472 			}
473 			break;
474 
475 		case PORT:
476 			if (isdigit(c)) {
477 				/* Save char to string */
478 				if (stringp ==
479 				    &string[sizeof (string) - 1]) {
480 					/* Would overflow, skip */
481 					parse = EOL;
482 				} else {
483 					/* Copy port digit char */
484 					*stringp++ = c;
485 				}
486 				break;
487 			} else if (c == '#' || isspace(c)) {
488 				/* End of port number, convert */
489 				*stringp = NULL;
490 				addrp->port = ntohs(atou(string));
491 
492 				/* End of parse, add entry */
493 				nl7c_addr_add(addrp);
494 				addrp = NULL;
495 				parse = EOL;
496 			} else {
497 				/* Unrecognized char, skip */
498 				parse = EOL;
499 				break;
500 			}
501 			if (c == '\n') {
502 				/* Found EOL, start on next line */
503 				parse = START;
504 			}
505 			break;
506 
507 		case EOL:
508 			if (c == '\n') {
509 				/* Found EOL, start on next line */
510 				parse = START;
511 			}
512 			break;
513 		}
514 
515 	}
516 	if (addrp != NULL) {
517 		kmem_free(addrp, sizeof (*addrp));
518 	}
519 	(void) VOP_CLOSE(vp, FREAD, 1, (offset_t)0, CRED(), NULL);
520 	VN_RELE(vp);
521 }
522 
523 /*
524  * Open and read each line from "/etc/nca/ncakmod.conf" and parse looking
525  * for the NCA enabled, the syntax is: status=enabled, all other lines will
526  * be ignored.
527  */
528 
529 static void
530 ncakmodconf_read(void)
531 {
532 	int	ret;
533 	struct vnode *vp;
534 	char	c;
535 	ssize_t resid;
536 	char	buf[1024];
537 	char	*ebp = &buf[sizeof (buf)];
538 	char	*bp = ebp;
539 	offset_t off = 0;
540 	enum parse_e {START, TOK, EOL} parse = START;
541 	char	*status = "status=enabled";
542 	char	*tok;
543 	char	*ncakmod = "/etc/nca/ncakmod.conf";
544 
545 	ret = vn_open(ncakmod, UIO_SYSSPACE, FREAD, 0, &vp, 0, 0);
546 	if (ret == ENOENT) {
547 		/* No ncakmod file, nothing to do */
548 		return;
549 	}
550 	if (ret != 0) {
551 		/* Error of some sort, tell'm about it */
552 		cmn_err(CE_WARN, "%s: open error %d", status, ret);
553 		return;
554 	}
555 	/*
556 	 * Read ncakmod one buf[] at a time, parse one char at a time.
557 	 */
558 	for (;;) {
559 		if (bp == ebp) {
560 			/* Nothing left in buf[], read another */
561 			ret = vn_rdwr(UIO_READ, vp, buf, sizeof (buf), off,
562 			    UIO_SYSSPACE, 0, (rlim64_t)0, CRED(), &resid);
563 			if (ret != 0) {
564 				/* Error of some sort, tell'm about it */
565 				cmn_err(CE_WARN, "%s: read error %d",
566 				    status, ret);
567 				break;
568 			}
569 			if (resid == sizeof (buf)) {
570 				/* EOF, done */
571 				break;
572 			}
573 			/* Initilize per buf[] state */
574 			bp = buf;
575 			ebp = &buf[sizeof (buf) - resid];
576 			off += sizeof (buf) - resid;
577 		}
578 		c = *bp++;
579 		switch (parse) {
580 		case START:
581 			/* Initilize all per file line state */
582 			tok = status;
583 			parse = TOK;
584 			/*FALLTHROUGH*/
585 		case TOK:
586 			if (c == '#') {
587 				/* Comment through end of line */
588 				parse = EOL;
589 				break;
590 			}
591 			if (isalpha(c) || c == '=') {
592 				if (c != *tok++) {
593 					/* Only know one token, skip */
594 					parse = EOL;
595 				}
596 			} else if (c == '\n') {
597 				/*
598 				 * Found EOL, if tok found done,
599 				 * else start on next-line.
600 				 */
601 				if (*tok == NULL) {
602 					nl7c_enabled = B_TRUE;
603 					goto done;
604 				}
605 				parse = START;
606 			} else {
607 				/* Unexpected char, skip */
608 				parse = EOL;
609 			}
610 			break;
611 
612 		case EOL:
613 			if (c == '\n') {
614 				/* Found EOL, start on next line */
615 				parse = START;
616 			}
617 			break;
618 		}
619 
620 	}
621 done:
622 	(void) VOP_CLOSE(vp, FREAD, 1, (offset_t)0, CRED(), NULL);
623 	VN_RELE(vp);
624 }
625 
626 /*
627  * Open and read each line from "/etc/nca/ncalogd.conf" and parse for
628  * the tokens and token text (i.e. key and value ncalogd.conf(4)):
629  *
630  *	status=enabled
631  *
632  *	logd_file_size=[0-9]+
633  *
634  *	logd_file_name=["]filename( filename)*["]
635  */
636 
637 static int	file_size = 1000000;
638 static caddr_t	fnv[NCA_FIOV_SZ];
639 
640 static void
641 ncalogdconf_read(void)
642 {
643 	int	ret;
644 	struct vnode *vp;
645 	char	c;
646 	int	sz;
647 	ssize_t resid;
648 	char	buf[1024];
649 	char	*ebp = &buf[sizeof (buf)];
650 	char	*bp = ebp;
651 	offset_t off = 0;
652 	enum parse_e {START, TOK, TEXT, EOL} parse = START;
653 	char	*tokstatus = "status\0enabled";
654 	char	*toksize = "logd_file_size";
655 	char	*tokfile = "logd_path_name";
656 	char	*tokstatusp;
657 	char	*toksizep;
658 	char	*tokfilep;
659 	char	*tok;
660 	int	tokdelim = 0;
661 	char	*ncalogd = "/etc/nca/ncalogd.conf";
662 	char	*ncadeflog = "/var/nca/log";
663 	char	file[TYPICALMAXPATHLEN] = {0};
664 	char	*fp = file;
665 	caddr_t	*fnvp = fnv;
666 
667 	ret = vn_open(ncalogd, UIO_SYSSPACE, FREAD, 0, &vp, 0, 0);
668 	if (ret == ENOENT) {
669 		/* No ncalogd file, nothing to do */
670 		return;
671 	}
672 	if (ret != 0) {
673 		/* Error of some sort, tell'm about it */
674 		cmn_err(CE_WARN, "ncalogdconf_read: %s: open error(%d).",
675 		    ncalogd, ret);
676 		return;
677 	}
678 	/*
679 	 * Read ncalogd.conf one buf[] at a time, parse one char at a time.
680 	 */
681 	for (;;) {
682 		if (bp == ebp) {
683 			/* Nothing left in buf[], read another */
684 			ret = vn_rdwr(UIO_READ, vp, buf, sizeof (buf), off,
685 			    UIO_SYSSPACE, 0, (rlim64_t)0, CRED(), &resid);
686 			if (ret != 0) {
687 				/* Error of some sort, tell'm about it */
688 				cmn_err(CE_WARN, "%s: read error %d",
689 				    ncalogd, ret);
690 				break;
691 			}
692 			if (resid == sizeof (buf)) {
693 				/* EOF, done */
694 				break;
695 			}
696 			/* Initilize per buf[] state */
697 			bp = buf;
698 			ebp = &buf[sizeof (buf) - resid];
699 			off += sizeof (buf) - resid;
700 		}
701 		c = *bp++;
702 		switch (parse) {
703 		case START:
704 			/* Initilize all per file line state */
705 			tokstatusp = tokstatus;
706 			toksizep = toksize;
707 			tokfilep = tokfile;
708 			tok = NULL;
709 			parse = TOK;
710 			sz = 0;
711 			/*FALLTHROUGH*/
712 		case TOK:
713 			if (isalpha(c) || c == '_') {
714 				/*
715 				 * Found a valid tok char, if matches
716 				 * any of the tokens continue else NULL
717 				 * then string pointer.
718 				 */
719 				if (tokstatusp != NULL && c != *tokstatusp++)
720 					tokstatusp = NULL;
721 				if (toksizep != NULL && c != *toksizep++)
722 					toksizep = NULL;
723 				if (tokfilep != NULL && c != *tokfilep++)
724 					tokfilep = NULL;
725 
726 				if (tokstatusp == NULL &&
727 				    toksizep == NULL &&
728 				    tokfilep == NULL) {
729 					/*
730 					 * All tok string pointers are NULL
731 					 * so skip rest of line.
732 					 */
733 					parse = EOL;
734 				}
735 			} else if (c == '=') {
736 				/*
737 				 * Found tok separator, if tok found get
738 				 * tok text, else skip rest of line.
739 				 */
740 				if (tokstatusp != NULL && *tokstatusp == NULL)
741 					tok = tokstatus;
742 				else if (toksizep != NULL && *toksizep == NULL)
743 					tok = toksize;
744 				else if (tokfilep != NULL && *tokfilep == NULL)
745 					tok = tokfile;
746 				if (tok != NULL)
747 					parse = TEXT;
748 				else
749 					parse = EOL;
750 			} else if (c == '\n') {
751 				/* Found EOL, start on next line */
752 				parse = START;
753 			} else {
754 				/* Comment or unknown char, skip rest of line */
755 				parse = EOL;
756 			}
757 			break;
758 		case TEXT:
759 			if (c == '\n') {
760 				/*
761 				 * Found EOL, finish up tok text processing
762 				 * (if any) and start on next line.
763 				 */
764 				if (tok == tokstatus) {
765 					if (*++tokstatusp == NULL)
766 						nl7c_logd_enabled = B_TRUE;
767 				} else if (tok == toksize) {
768 					file_size = sz;
769 				} else if (tok == tokfile) {
770 					if (tokdelim == 0) {
771 						/* Non delimited path name */
772 						*fnvp++ = strdup(file);
773 					} else if (fp != file) {
774 						/* No closing delimiter */
775 						/*EMPTY*/;
776 					}
777 				}
778 				parse = START;
779 			} else if (tok == tokstatus) {
780 				if (! isalpha(c) || *++tokstatusp == NULL ||
781 				    c != *tokstatusp) {
782 					/* Not enabled, skip line */
783 					parse = EOL;
784 				}
785 			} else if (tok == toksize) {
786 				if (isdigit(c)) {
787 					sz *= 10;
788 					sz += c - '0';
789 				} else {
790 					/* Not a decimal digit, skip line */
791 					parse = EOL;
792 				}
793 			} else {
794 				/* File name */
795 				if (c == '"' && tokdelim++ == 0) {
796 					/* Opening delimiter, skip */
797 					/*EMPTY*/;
798 				} else if (c == '"' || c == ' ') {
799 					/* List delim or filename separator */
800 					*fnvp++ = strdup(file);
801 					fp = file;
802 				} else if (fp < &file[sizeof (file) - 1]) {
803 					/* Filename char */
804 					*fp++ = c;
805 				} else {
806 					/* Filename to long, skip line */
807 					parse = EOL;
808 				}
809 			}
810 			break;
811 
812 		case EOL:
813 			if (c == '\n') {
814 				/* Found EOL, start on next line */
815 				parse = START;
816 			}
817 			break;
818 		}
819 
820 	}
821 done:
822 	(void) VOP_CLOSE(vp, FREAD, 1, (offset_t)0, CRED(), NULL);
823 	VN_RELE(vp);
824 
825 	if (nl7c_logd_enabled) {
826 		if (fnvp == fnv) {
827 			/*
828 			 * No logfile was specified and found so
829 			 * so use defualt NCA log file path.
830 			 */
831 			*fnvp++ = strdup(ncadeflog);
832 		}
833 		if (fnvp < &fnv[NCA_FIOV_SZ]) {
834 			/* NULL terminate list */
835 			*fnvp = NULL;
836 		}
837 	}
838 }
839 
840 void
841 nl7clogd_startup(void)
842 {
843 	static kmutex_t startup;
844 
845 	/*
846 	 * Called on the first log() attempt, have to wait until then to
847 	 * initialize logd as at logdconf_read() the root fs is read-only.
848 	 */
849 	mutex_enter(&startup);
850 	if (nl7c_logd_started) {
851 		/* Lost the race, nothing todo */
852 		mutex_exit(&startup);
853 		return;
854 	}
855 	nl7c_logd_started = B_TRUE;
856 	if (! nl7c_logd_init(file_size, fnv)) {
857 		/* Failure, disable logging */
858 		nl7c_logd_enabled = B_FALSE;
859 		cmn_err(CE_WARN, "nl7clogd_startup: failed, disabling loggin");
860 		mutex_exit(&startup);
861 		return;
862 	}
863 	mutex_exit(&startup);
864 }
865 
866 
867 void
868 nl7c_startup()
869 {
870 	/*
871 	 * Open, read, and parse the NCA logd configuration file,
872 	 * then initialize URI processing and NCA compat.
873 	 */
874 	ncalogdconf_read();
875 	nl7c_uri_init();
876 	nl7c_nca_init();
877 }
878 
879 void
880 nl7c_init()
881 {
882 	/* Open, read, and parse the NCA kmod configuration file */
883 	ncakmodconf_read();
884 
885 	if (nl7c_enabled) {
886 		/*
887 		 * NL7C is enabled so open, read, and parse
888 		 * the NCA address/port configuration file
889 		 * and call startup() to finish config/init.
890 		 */
891 		ncaportconf_read();
892 		nl7c_startup();
893 	}
894 }
895 
896 /*
897  * The main processing function called by accept() on a newly created
898  * socket prior to returning it to the caller of accept().
899  *
900  * Here data is read from the socket until a completed L7 request parse
901  * is completed. Data will be read in the context of the user thread
902  * which called accept(), when parse has been completed either B_TRUE
903  * or B_FALSE will be returned.
904  *
905  * If NL7C successfully process the L7 protocol request, i.e. generates
906  * a response, B_TRUE will be returned.
907  *
908  * Else, B_FALSE will be returned if NL7C can't process the request:
909  *
910  * 1) Couldn't locate a URI within the request.
911  *
912  * 2) URI scheme not reqcognized.
913  *
914  * 3) A request which can't be processed.
915  *
916  * 4) A request which could be processed but NL7C dosen't currently have
917  *    the response data. In which case NL7C will parse the returned response
918  *    from the application for possible caching for subsequent request(s).
919  */
920 
921 volatile uint64_t nl7c_proc_cnt = 0;
922 volatile uint64_t nl7c_proc_error = 0;
923 volatile uint64_t nl7c_proc_ETIME = 0;
924 volatile uint64_t nl7c_proc_again = 0;
925 volatile uint64_t nl7c_proc_next = 0;
926 volatile uint64_t nl7c_proc_rcv = 0;
927 volatile uint64_t nl7c_proc_noLRI = 0;
928 volatile uint64_t nl7c_proc_nodata = 0;
929 volatile uint64_t nl7c_proc_parse = 0;
930 
931 boolean_t
932 nl7c_process(struct sonode *so, boolean_t nonblocking)
933 {
934 	vnode_t	*vp = SOTOV(so);
935 	sotpi_info_t *sti = SOTOTPI(so);
936 	mblk_t	*rmp = sti->sti_nl7c_rcv_mp;
937 	clock_t	timout;
938 	rval_t	rval;
939 	uchar_t pri;
940 	int 	pflag;
941 	int	error;
942 	boolean_t more;
943 	boolean_t ret = B_FALSE;
944 	boolean_t first = B_TRUE;
945 	boolean_t pollin = (sti->sti_nl7c_flags & NL7C_POLLIN);
946 
947 	nl7c_proc_cnt++;
948 
949 	/* Caller has so_lock enter()ed */
950 	error = so_lock_read_intr(so, nonblocking ? FNDELAY|FNONBLOCK : 0);
951 	if (error) {
952 		/* Couldn't read lock, pass on this socket */
953 		sti->sti_nl7c_flags = 0;
954 		nl7c_proc_noLRI++;
955 		return (B_FALSE);
956 	}
957 	/* Exit so_lock for now, will be reenter()ed prior to return */
958 	mutex_exit(&so->so_lock);
959 
960 	if (pollin)
961 		sti->sti_nl7c_flags &= ~NL7C_POLLIN;
962 
963 	/* Initialize some kstrgetmsg() constants */
964 	pflag = MSG_ANY | MSG_DELAYERROR;
965 	pri = 0;
966 	if (nonblocking) {
967 		/* Non blocking so don't block */
968 		timout = 0;
969 	} else if (sti->sti_nl7c_flags & NL7C_SOPERSIST) {
970 		/* 2nd or more time(s) here so use keep-alive value */
971 		timout = nca_http_keep_alive_timeout;
972 	} else {
973 		/* 1st time here so use connection value */
974 		timout = nca_http_timeout;
975 	}
976 
977 	rval.r_vals = 0;
978 	do {
979 		/*
980 		 * First time through, if no data left over from a previous
981 		 * kstrgetmsg() then try to get some, else just process it.
982 		 *
983 		 * Thereafter, rmp = NULL after the successful kstrgetmsg()
984 		 * so try to get some new data and append to list (i.e. until
985 		 * enough fragments are collected for a successful parse).
986 		 */
987 		if (rmp == NULL) {
988 
989 			error = kstrgetmsg(vp, &rmp, NULL, &pri, &pflag,
990 			    timout, &rval);
991 			if (error) {
992 				if (error == ETIME) {
993 					/* Timeout */
994 					nl7c_proc_ETIME++;
995 				} else if (error != EWOULDBLOCK) {
996 					/* Error of some sort */
997 					nl7c_proc_error++;
998 					rval.r_v.r_v2 = error;
999 					sti->sti_nl7c_flags = 0;
1000 					break;
1001 				}
1002 				error = 0;
1003 			}
1004 			if (rmp != NULL) {
1005 				mblk_t	*mp = sti->sti_nl7c_rcv_mp;
1006 
1007 
1008 				if (mp == NULL) {
1009 					/* Just new data, common case */
1010 					sti->sti_nl7c_rcv_mp = rmp;
1011 				} else {
1012 					/* Add new data to tail */
1013 					while (mp->b_cont != NULL)
1014 						mp = mp->b_cont;
1015 					mp->b_cont = rmp;
1016 				}
1017 			}
1018 			if (sti->sti_nl7c_rcv_mp == NULL) {
1019 				/* No data */
1020 				nl7c_proc_nodata++;
1021 				if (timout > 0 || (first && pollin)) {
1022 					/* Expected data so EOF */
1023 					ret = B_TRUE;
1024 				} else if (sti->sti_nl7c_flags &
1025 				    NL7C_SOPERSIST) {
1026 					/* Persistent so just checking */
1027 					ret = B_FALSE;
1028 				}
1029 				break;
1030 			}
1031 			rmp = NULL;
1032 		}
1033 		first = B_FALSE;
1034 	again:
1035 		nl7c_proc_parse++;
1036 
1037 		more = nl7c_parse(so, nonblocking, &ret);
1038 
1039 		if (ret == B_TRUE && (sti->sti_nl7c_flags & NL7C_SOPERSIST)) {
1040 			/*
1041 			 * Parse complete, cache hit, response on its way,
1042 			 * socket is persistent so try to process the next
1043 			 * request.
1044 			 */
1045 			if (nonblocking) {
1046 				ret = B_FALSE;
1047 				break;
1048 			}
1049 			if (sti->sti_nl7c_rcv_mp) {
1050 				/* More recv-side data, pipelined */
1051 				nl7c_proc_again++;
1052 				goto again;
1053 			}
1054 			nl7c_proc_next++;
1055 			if (nonblocking)
1056 				timout = 0;
1057 			else
1058 				timout = nca_http_keep_alive_timeout;
1059 
1060 			more = B_TRUE;
1061 		}
1062 
1063 	} while (more);
1064 
1065 	if (sti->sti_nl7c_rcv_mp) {
1066 		nl7c_proc_rcv++;
1067 	}
1068 	sti->sti_nl7c_rcv_rval = rval.r_vals;
1069 	/* Renter so_lock, caller called with it enter()ed */
1070 	mutex_enter(&so->so_lock);
1071 	so_unlock_read(so);
1072 
1073 	return (ret);
1074 }
1075