1*3b2bd0f6Slogwang /*
2*3b2bd0f6Slogwang * debug.c
3*3b2bd0f6Slogwang *
4*3b2bd0f6Slogwang * Copyright (c) 1996-1999 Whistle Communications, Inc.
5*3b2bd0f6Slogwang * All rights reserved.
6*3b2bd0f6Slogwang *
7*3b2bd0f6Slogwang * Subject to the following obligations and disclaimer of warranty, use and
8*3b2bd0f6Slogwang * redistribution of this software, in source or object code forms, with or
9*3b2bd0f6Slogwang * without modifications are expressly permitted by Whistle Communications;
10*3b2bd0f6Slogwang * provided, however, that:
11*3b2bd0f6Slogwang * 1. Any and all reproductions of the source or object code must include the
12*3b2bd0f6Slogwang * copyright notice above and the following disclaimer of warranties; and
13*3b2bd0f6Slogwang * 2. No rights are granted, in any manner or form, to use Whistle
14*3b2bd0f6Slogwang * Communications, Inc. trademarks, including the mark "WHISTLE
15*3b2bd0f6Slogwang * COMMUNICATIONS" on advertising, endorsements, or otherwise except as
16*3b2bd0f6Slogwang * such appears in the above copyright notice or in the software.
17*3b2bd0f6Slogwang *
18*3b2bd0f6Slogwang * THIS SOFTWARE IS BEING PROVIDED BY WHISTLE COMMUNICATIONS "AS IS", AND
19*3b2bd0f6Slogwang * TO THE MAXIMUM EXTENT PERMITTED BY LAW, WHISTLE COMMUNICATIONS MAKES NO
20*3b2bd0f6Slogwang * REPRESENTATIONS OR WARRANTIES, EXPRESS OR IMPLIED, REGARDING THIS SOFTWARE,
21*3b2bd0f6Slogwang * INCLUDING WITHOUT LIMITATION, ANY AND ALL IMPLIED WARRANTIES OF
22*3b2bd0f6Slogwang * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, OR NON-INFRINGEMENT.
23*3b2bd0f6Slogwang * WHISTLE COMMUNICATIONS DOES NOT WARRANT, GUARANTEE, OR MAKE ANY
24*3b2bd0f6Slogwang * REPRESENTATIONS REGARDING THE USE OF, OR THE RESULTS OF THE USE OF THIS
25*3b2bd0f6Slogwang * SOFTWARE IN TERMS OF ITS CORRECTNESS, ACCURACY, RELIABILITY OR OTHERWISE.
26*3b2bd0f6Slogwang * IN NO EVENT SHALL WHISTLE COMMUNICATIONS BE LIABLE FOR ANY DAMAGES
27*3b2bd0f6Slogwang * RESULTING FROM OR ARISING OUT OF ANY USE OF THIS SOFTWARE, INCLUDING
28*3b2bd0f6Slogwang * WITHOUT LIMITATION, ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY,
29*3b2bd0f6Slogwang * PUNITIVE, OR CONSEQUENTIAL DAMAGES, PROCUREMENT OF SUBSTITUTE GOODS OR
30*3b2bd0f6Slogwang * SERVICES, LOSS OF USE, DATA OR PROFITS, HOWEVER CAUSED AND UNDER ANY
31*3b2bd0f6Slogwang * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
32*3b2bd0f6Slogwang * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
33*3b2bd0f6Slogwang * THIS SOFTWARE, EVEN IF WHISTLE COMMUNICATIONS IS ADVISED OF THE POSSIBILITY
34*3b2bd0f6Slogwang * OF SUCH DAMAGE.
35*3b2bd0f6Slogwang *
36*3b2bd0f6Slogwang * Author: Archie Cobbs <[email protected]>
37*3b2bd0f6Slogwang *
38*3b2bd0f6Slogwang * $Whistle: debug.c,v 1.24 1999/01/24 01:15:33 archie Exp $
39*3b2bd0f6Slogwang */
40*3b2bd0f6Slogwang
41*3b2bd0f6Slogwang #include <sys/cdefs.h>
42*3b2bd0f6Slogwang __FBSDID("$FreeBSD$");
43*3b2bd0f6Slogwang
44*3b2bd0f6Slogwang #include <sys/types.h>
45*3b2bd0f6Slogwang #include <sys/time.h>
46*3b2bd0f6Slogwang #include <sys/ioctl.h>
47*3b2bd0f6Slogwang
48*3b2bd0f6Slogwang #include <stdarg.h>
49*3b2bd0f6Slogwang
50*3b2bd0f6Slogwang #include <netinet/in.h>
51*3b2bd0f6Slogwang #include <net/ethernet.h>
52*3b2bd0f6Slogwang #include <net/bpf.h>
53*3b2bd0f6Slogwang
54*3b2bd0f6Slogwang #include <netgraph/ng_message.h>
55*3b2bd0f6Slogwang #include <netgraph/ng_socket.h>
56*3b2bd0f6Slogwang
57*3b2bd0f6Slogwang #include "netgraph.h"
58*3b2bd0f6Slogwang #include "internal.h"
59*3b2bd0f6Slogwang
60*3b2bd0f6Slogwang #include <netgraph/ng_UI.h>
61*3b2bd0f6Slogwang #include <netgraph/ng_async.h>
62*3b2bd0f6Slogwang #include <netgraph/ng_atmllc.h>
63*3b2bd0f6Slogwang #include <netgraph/ng_bpf.h>
64*3b2bd0f6Slogwang #include <netgraph/ng_bridge.h>
65*3b2bd0f6Slogwang #include <netgraph/ng_car.h>
66*3b2bd0f6Slogwang #include <netgraph/ng_cisco.h>
67*3b2bd0f6Slogwang #include <netgraph/ng_deflate.h>
68*3b2bd0f6Slogwang #include <netgraph/ng_device.h>
69*3b2bd0f6Slogwang #include <netgraph/ng_echo.h>
70*3b2bd0f6Slogwang #include <netgraph/ng_eiface.h>
71*3b2bd0f6Slogwang #include <netgraph/ng_etf.h>
72*3b2bd0f6Slogwang #include <netgraph/ng_ether.h>
73*3b2bd0f6Slogwang #include <netgraph/ng_ether_echo.h>
74*3b2bd0f6Slogwang #include <netgraph/ng_frame_relay.h>
75*3b2bd0f6Slogwang #include <netgraph/ng_gif.h>
76*3b2bd0f6Slogwang #include <netgraph/ng_gif_demux.h>
77*3b2bd0f6Slogwang #include <netgraph/ng_hole.h>
78*3b2bd0f6Slogwang #include <netgraph/ng_hub.h>
79*3b2bd0f6Slogwang #include <netgraph/ng_iface.h>
80*3b2bd0f6Slogwang #include <netgraph/ng_ip_input.h>
81*3b2bd0f6Slogwang #include <netgraph/ng_ipfw.h>
82*3b2bd0f6Slogwang #include <netgraph/ng_ksocket.h>
83*3b2bd0f6Slogwang #include <netgraph/ng_l2tp.h>
84*3b2bd0f6Slogwang #include <netgraph/ng_lmi.h>
85*3b2bd0f6Slogwang #include <netgraph/ng_mppc.h>
86*3b2bd0f6Slogwang #include <netgraph/ng_nat.h>
87*3b2bd0f6Slogwang #include <netgraph/netflow/ng_netflow.h>
88*3b2bd0f6Slogwang #include <netgraph/ng_one2many.h>
89*3b2bd0f6Slogwang #include <netgraph/ng_patch.h>
90*3b2bd0f6Slogwang #include <netgraph/ng_pipe.h>
91*3b2bd0f6Slogwang #include <netgraph/ng_ppp.h>
92*3b2bd0f6Slogwang #include <netgraph/ng_pppoe.h>
93*3b2bd0f6Slogwang #include <netgraph/ng_pptpgre.h>
94*3b2bd0f6Slogwang #include <netgraph/ng_pred1.h>
95*3b2bd0f6Slogwang #include <netgraph/ng_rfc1490.h>
96*3b2bd0f6Slogwang #include <netgraph/ng_socket.h>
97*3b2bd0f6Slogwang #include <netgraph/ng_source.h>
98*3b2bd0f6Slogwang #include <netgraph/ng_split.h>
99*3b2bd0f6Slogwang #include <netgraph/ng_sppp.h>
100*3b2bd0f6Slogwang #include <netgraph/ng_tag.h>
101*3b2bd0f6Slogwang #include <netgraph/ng_tcpmss.h>
102*3b2bd0f6Slogwang #include <netgraph/ng_tee.h>
103*3b2bd0f6Slogwang #include <netgraph/ng_tty.h>
104*3b2bd0f6Slogwang #include <netgraph/ng_vjc.h>
105*3b2bd0f6Slogwang #include <netgraph/ng_vlan.h>
106*3b2bd0f6Slogwang #ifdef WHISTLE
107*3b2bd0f6Slogwang #include <machine/../isa/df_def.h>
108*3b2bd0f6Slogwang #include <machine/../isa/if_wfra.h>
109*3b2bd0f6Slogwang #include <machine/../isa/ipac.h>
110*3b2bd0f6Slogwang #include <netgraph/ng_df.h>
111*3b2bd0f6Slogwang #include <netgraph/ng_ipac.h>
112*3b2bd0f6Slogwang #include <netgraph/ng_tn.h>
113*3b2bd0f6Slogwang #endif
114*3b2bd0f6Slogwang
115*3b2bd0f6Slogwang /* Global debug level */
116*3b2bd0f6Slogwang int _gNgDebugLevel = 0;
117*3b2bd0f6Slogwang
118*3b2bd0f6Slogwang /* Debug printing functions */
119*3b2bd0f6Slogwang void (*_NgLog) (const char *fmt,...) = warn;
120*3b2bd0f6Slogwang void (*_NgLogx) (const char *fmt,...) = warnx;
121*3b2bd0f6Slogwang
122*3b2bd0f6Slogwang /* Internal functions */
123*3b2bd0f6Slogwang static const char *NgCookie(int cookie);
124*3b2bd0f6Slogwang
125*3b2bd0f6Slogwang /* Known typecookie list */
126*3b2bd0f6Slogwang struct ng_cookie {
127*3b2bd0f6Slogwang int cookie;
128*3b2bd0f6Slogwang const char *type;
129*3b2bd0f6Slogwang };
130*3b2bd0f6Slogwang
131*3b2bd0f6Slogwang #define COOKIE(c) { NGM_ ## c ## _COOKIE, #c }
132*3b2bd0f6Slogwang
133*3b2bd0f6Slogwang /* List of known cookies */
134*3b2bd0f6Slogwang static const struct ng_cookie cookies[] = {
135*3b2bd0f6Slogwang COOKIE(UI),
136*3b2bd0f6Slogwang COOKIE(ASYNC),
137*3b2bd0f6Slogwang COOKIE(ATMLLC),
138*3b2bd0f6Slogwang COOKIE(BPF),
139*3b2bd0f6Slogwang COOKIE(BRIDGE),
140*3b2bd0f6Slogwang COOKIE(CAR),
141*3b2bd0f6Slogwang COOKIE(CISCO),
142*3b2bd0f6Slogwang COOKIE(DEFLATE),
143*3b2bd0f6Slogwang COOKIE(DEVICE),
144*3b2bd0f6Slogwang COOKIE(ECHO),
145*3b2bd0f6Slogwang COOKIE(EIFACE),
146*3b2bd0f6Slogwang COOKIE(ETF),
147*3b2bd0f6Slogwang COOKIE(ETHER),
148*3b2bd0f6Slogwang COOKIE(ETHER_ECHO),
149*3b2bd0f6Slogwang COOKIE(FRAMERELAY),
150*3b2bd0f6Slogwang COOKIE(GIF),
151*3b2bd0f6Slogwang COOKIE(GIF_DEMUX),
152*3b2bd0f6Slogwang COOKIE(GENERIC),
153*3b2bd0f6Slogwang COOKIE(HOLE),
154*3b2bd0f6Slogwang COOKIE(HUB),
155*3b2bd0f6Slogwang COOKIE(IFACE),
156*3b2bd0f6Slogwang COOKIE(IP_INPUT),
157*3b2bd0f6Slogwang COOKIE(IPFW),
158*3b2bd0f6Slogwang COOKIE(KSOCKET),
159*3b2bd0f6Slogwang COOKIE(L2TP),
160*3b2bd0f6Slogwang COOKIE(LMI),
161*3b2bd0f6Slogwang COOKIE(MPPC),
162*3b2bd0f6Slogwang COOKIE(NAT),
163*3b2bd0f6Slogwang COOKIE(NETFLOW),
164*3b2bd0f6Slogwang COOKIE(ONE2MANY),
165*3b2bd0f6Slogwang COOKIE(PATCH),
166*3b2bd0f6Slogwang COOKIE(PIPE),
167*3b2bd0f6Slogwang COOKIE(PPP),
168*3b2bd0f6Slogwang COOKIE(PPPOE),
169*3b2bd0f6Slogwang COOKIE(PPTPGRE),
170*3b2bd0f6Slogwang COOKIE(PRED1),
171*3b2bd0f6Slogwang COOKIE(RFC1490),
172*3b2bd0f6Slogwang COOKIE(SOCKET),
173*3b2bd0f6Slogwang COOKIE(SOURCE),
174*3b2bd0f6Slogwang COOKIE(SPLIT),
175*3b2bd0f6Slogwang COOKIE(SPPP),
176*3b2bd0f6Slogwang COOKIE(TAG),
177*3b2bd0f6Slogwang COOKIE(TCPMSS),
178*3b2bd0f6Slogwang COOKIE(TEE),
179*3b2bd0f6Slogwang COOKIE(TTY),
180*3b2bd0f6Slogwang COOKIE(VJC),
181*3b2bd0f6Slogwang COOKIE(VLAN),
182*3b2bd0f6Slogwang #ifdef WHISTLE
183*3b2bd0f6Slogwang COOKIE(DF),
184*3b2bd0f6Slogwang COOKIE(IPAC),
185*3b2bd0f6Slogwang COOKIE(TN),
186*3b2bd0f6Slogwang COOKIE(WFRA),
187*3b2bd0f6Slogwang #endif
188*3b2bd0f6Slogwang { 0, NULL }
189*3b2bd0f6Slogwang };
190*3b2bd0f6Slogwang
191*3b2bd0f6Slogwang /*
192*3b2bd0f6Slogwang * Set debug level, ie, verbosity, if "level" is non-negative.
193*3b2bd0f6Slogwang * Returns old debug level.
194*3b2bd0f6Slogwang */
195*3b2bd0f6Slogwang int
NgSetDebug(int level)196*3b2bd0f6Slogwang NgSetDebug(int level)
197*3b2bd0f6Slogwang {
198*3b2bd0f6Slogwang int old = _gNgDebugLevel;
199*3b2bd0f6Slogwang
200*3b2bd0f6Slogwang if (level >= 0)
201*3b2bd0f6Slogwang _gNgDebugLevel = level;
202*3b2bd0f6Slogwang return (old);
203*3b2bd0f6Slogwang }
204*3b2bd0f6Slogwang
205*3b2bd0f6Slogwang /*
206*3b2bd0f6Slogwang * Set debug logging functions.
207*3b2bd0f6Slogwang */
208*3b2bd0f6Slogwang void
NgSetErrLog(void (* log)(const char * fmt,...),void (* logx)(const char * fmt,...))209*3b2bd0f6Slogwang NgSetErrLog(void (*log) (const char *fmt,...),
210*3b2bd0f6Slogwang void (*logx) (const char *fmt,...))
211*3b2bd0f6Slogwang {
212*3b2bd0f6Slogwang _NgLog = log;
213*3b2bd0f6Slogwang _NgLogx = logx;
214*3b2bd0f6Slogwang }
215*3b2bd0f6Slogwang
216*3b2bd0f6Slogwang /*
217*3b2bd0f6Slogwang * Display a netgraph sockaddr
218*3b2bd0f6Slogwang */
219*3b2bd0f6Slogwang void
_NgDebugSockaddr(const struct sockaddr_ng * sg)220*3b2bd0f6Slogwang _NgDebugSockaddr(const struct sockaddr_ng *sg)
221*3b2bd0f6Slogwang {
222*3b2bd0f6Slogwang NGLOGX("SOCKADDR: { fam=%d len=%d addr=\"%s\" }",
223*3b2bd0f6Slogwang sg->sg_family, sg->sg_len, sg->sg_data);
224*3b2bd0f6Slogwang }
225*3b2bd0f6Slogwang
226*3b2bd0f6Slogwang #define ARGS_BUFSIZE 2048
227*3b2bd0f6Slogwang #define RECURSIVE_DEBUG_ADJUST 4
228*3b2bd0f6Slogwang
229*3b2bd0f6Slogwang /*
230*3b2bd0f6Slogwang * Display a negraph message
231*3b2bd0f6Slogwang */
232*3b2bd0f6Slogwang void
_NgDebugMsg(const struct ng_mesg * msg,const char * path)233*3b2bd0f6Slogwang _NgDebugMsg(const struct ng_mesg *msg, const char *path)
234*3b2bd0f6Slogwang {
235*3b2bd0f6Slogwang u_char buf[2 * sizeof(struct ng_mesg) + ARGS_BUFSIZE];
236*3b2bd0f6Slogwang struct ng_mesg *const req = (struct ng_mesg *)buf;
237*3b2bd0f6Slogwang struct ng_mesg *const bin = (struct ng_mesg *)req->data;
238*3b2bd0f6Slogwang int arglen, csock = -1;
239*3b2bd0f6Slogwang
240*3b2bd0f6Slogwang /* Display header stuff */
241*3b2bd0f6Slogwang NGLOGX("NG_MESG :");
242*3b2bd0f6Slogwang NGLOGX(" vers %d", msg->header.version);
243*3b2bd0f6Slogwang NGLOGX(" arglen %u", msg->header.arglen);
244*3b2bd0f6Slogwang NGLOGX(" flags %x", msg->header.flags);
245*3b2bd0f6Slogwang NGLOGX(" token %u", msg->header.token);
246*3b2bd0f6Slogwang NGLOGX(" cookie %s (%u)",
247*3b2bd0f6Slogwang NgCookie(msg->header.typecookie), msg->header.typecookie);
248*3b2bd0f6Slogwang
249*3b2bd0f6Slogwang /* At lower debugging levels, skip ASCII translation */
250*3b2bd0f6Slogwang if (_gNgDebugLevel <= 2)
251*3b2bd0f6Slogwang goto fail2;
252*3b2bd0f6Slogwang
253*3b2bd0f6Slogwang /* If path is not absolute, don't bother trying to use relative
254*3b2bd0f6Slogwang address on a different socket for the ASCII translation */
255*3b2bd0f6Slogwang if (strchr(path, ':') == NULL)
256*3b2bd0f6Slogwang goto fail2;
257*3b2bd0f6Slogwang
258*3b2bd0f6Slogwang /* Get a temporary socket */
259*3b2bd0f6Slogwang if (NgMkSockNode(NULL, &csock, NULL) < 0)
260*3b2bd0f6Slogwang goto fail;
261*3b2bd0f6Slogwang
262*3b2bd0f6Slogwang /* Copy binary message into request message payload */
263*3b2bd0f6Slogwang arglen = msg->header.arglen;
264*3b2bd0f6Slogwang if (arglen > ARGS_BUFSIZE)
265*3b2bd0f6Slogwang arglen = ARGS_BUFSIZE;
266*3b2bd0f6Slogwang memcpy(bin, msg, sizeof(*msg) + arglen);
267*3b2bd0f6Slogwang bin->header.arglen = arglen;
268*3b2bd0f6Slogwang
269*3b2bd0f6Slogwang /* Lower debugging to avoid infinite recursion */
270*3b2bd0f6Slogwang _gNgDebugLevel -= RECURSIVE_DEBUG_ADJUST;
271*3b2bd0f6Slogwang
272*3b2bd0f6Slogwang /* Ask the node to translate the binary message to ASCII for us */
273*3b2bd0f6Slogwang if (NgSendMsg(csock, path, NGM_GENERIC_COOKIE,
274*3b2bd0f6Slogwang NGM_BINARY2ASCII, bin, sizeof(*bin) + bin->header.arglen) < 0) {
275*3b2bd0f6Slogwang _gNgDebugLevel += RECURSIVE_DEBUG_ADJUST;
276*3b2bd0f6Slogwang goto fail;
277*3b2bd0f6Slogwang }
278*3b2bd0f6Slogwang if (NgRecvMsg(csock, req, sizeof(buf), NULL) < 0) {
279*3b2bd0f6Slogwang _gNgDebugLevel += RECURSIVE_DEBUG_ADJUST;
280*3b2bd0f6Slogwang goto fail;
281*3b2bd0f6Slogwang }
282*3b2bd0f6Slogwang
283*3b2bd0f6Slogwang /* Restore debugging level */
284*3b2bd0f6Slogwang _gNgDebugLevel += RECURSIVE_DEBUG_ADJUST;
285*3b2bd0f6Slogwang
286*3b2bd0f6Slogwang /* Display command string and arguments */
287*3b2bd0f6Slogwang NGLOGX(" cmd %s (%d)", bin->header.cmdstr, bin->header.cmd);
288*3b2bd0f6Slogwang NGLOGX(" args %s", bin->data);
289*3b2bd0f6Slogwang goto done;
290*3b2bd0f6Slogwang
291*3b2bd0f6Slogwang fail:
292*3b2bd0f6Slogwang /* Just display binary version */
293*3b2bd0f6Slogwang NGLOGX(" [error decoding message: %s]", strerror(errno));
294*3b2bd0f6Slogwang fail2:
295*3b2bd0f6Slogwang NGLOGX(" cmd %d", msg->header.cmd);
296*3b2bd0f6Slogwang NGLOGX(" args (%d bytes)", msg->header.arglen);
297*3b2bd0f6Slogwang _NgDebugBytes((u_char *)msg->data, msg->header.arglen);
298*3b2bd0f6Slogwang
299*3b2bd0f6Slogwang done:
300*3b2bd0f6Slogwang if (csock != -1)
301*3b2bd0f6Slogwang (void)close(csock);
302*3b2bd0f6Slogwang }
303*3b2bd0f6Slogwang
304*3b2bd0f6Slogwang /*
305*3b2bd0f6Slogwang * Return the name of the node type corresponding to the cookie
306*3b2bd0f6Slogwang */
307*3b2bd0f6Slogwang static const char *
NgCookie(int cookie)308*3b2bd0f6Slogwang NgCookie(int cookie)
309*3b2bd0f6Slogwang {
310*3b2bd0f6Slogwang int k;
311*3b2bd0f6Slogwang
312*3b2bd0f6Slogwang for (k = 0; cookies[k].cookie != 0; k++) {
313*3b2bd0f6Slogwang if (cookies[k].cookie == cookie)
314*3b2bd0f6Slogwang return cookies[k].type;
315*3b2bd0f6Slogwang }
316*3b2bd0f6Slogwang return "??";
317*3b2bd0f6Slogwang }
318*3b2bd0f6Slogwang
319*3b2bd0f6Slogwang /*
320*3b2bd0f6Slogwang * Dump bytes in hex
321*3b2bd0f6Slogwang */
322*3b2bd0f6Slogwang void
_NgDebugBytes(const u_char * ptr,int len)323*3b2bd0f6Slogwang _NgDebugBytes(const u_char *ptr, int len)
324*3b2bd0f6Slogwang {
325*3b2bd0f6Slogwang char buf[100];
326*3b2bd0f6Slogwang int k, count;
327*3b2bd0f6Slogwang
328*3b2bd0f6Slogwang #define BYPERLINE 16
329*3b2bd0f6Slogwang
330*3b2bd0f6Slogwang for (count = 0; count < len; ptr += BYPERLINE, count += BYPERLINE) {
331*3b2bd0f6Slogwang
332*3b2bd0f6Slogwang /* Do hex */
333*3b2bd0f6Slogwang snprintf(buf, sizeof(buf), "%04x: ", count);
334*3b2bd0f6Slogwang for (k = 0; k < BYPERLINE; k++, count++)
335*3b2bd0f6Slogwang if (count < len)
336*3b2bd0f6Slogwang snprintf(buf + strlen(buf),
337*3b2bd0f6Slogwang sizeof(buf) - strlen(buf), "%02x ", ptr[k]);
338*3b2bd0f6Slogwang else
339*3b2bd0f6Slogwang snprintf(buf + strlen(buf),
340*3b2bd0f6Slogwang sizeof(buf) - strlen(buf), " ");
341*3b2bd0f6Slogwang snprintf(buf + strlen(buf), sizeof(buf) - strlen(buf), " ");
342*3b2bd0f6Slogwang count -= BYPERLINE;
343*3b2bd0f6Slogwang
344*3b2bd0f6Slogwang /* Do ASCII */
345*3b2bd0f6Slogwang for (k = 0; k < BYPERLINE; k++, count++)
346*3b2bd0f6Slogwang if (count < len)
347*3b2bd0f6Slogwang snprintf(buf + strlen(buf),
348*3b2bd0f6Slogwang sizeof(buf) - strlen(buf),
349*3b2bd0f6Slogwang "%c", isprint(ptr[k]) ? ptr[k] : '.');
350*3b2bd0f6Slogwang else
351*3b2bd0f6Slogwang snprintf(buf + strlen(buf),
352*3b2bd0f6Slogwang sizeof(buf) - strlen(buf), " ");
353*3b2bd0f6Slogwang count -= BYPERLINE;
354*3b2bd0f6Slogwang
355*3b2bd0f6Slogwang /* Print it */
356*3b2bd0f6Slogwang NGLOGX("%s", buf);
357*3b2bd0f6Slogwang }
358*3b2bd0f6Slogwang }
359*3b2bd0f6Slogwang
360