xref: /freebsd/lib/libc/stdio/xprintf_quote.c (revision b3e7694832e81d7a904a10f525f8797b753bf0d3)
16dbacee2SPoul-Henning Kamp /*-
2*4d846d26SWarner Losh  * SPDX-License-Identifier: BSD-2-Clause
3d915a14eSPedro F. Giffuni  *
46dbacee2SPoul-Henning Kamp  * Copyright (c) 2005 Poul-Henning Kamp
56dbacee2SPoul-Henning Kamp  * All rights reserved.
66dbacee2SPoul-Henning Kamp  *
76dbacee2SPoul-Henning Kamp  * Redistribution and use in source and binary forms, with or without
86dbacee2SPoul-Henning Kamp  * modification, are permitted provided that the following conditions
96dbacee2SPoul-Henning Kamp  * are met:
106dbacee2SPoul-Henning Kamp  * 1. Redistributions of source code must retain the above copyright
116dbacee2SPoul-Henning Kamp  *    notice, this list of conditions and the following disclaimer.
126dbacee2SPoul-Henning Kamp  * 2. Redistributions in binary form must reproduce the above copyright
136dbacee2SPoul-Henning Kamp  *    notice, this list of conditions and the following disclaimer in the
146dbacee2SPoul-Henning Kamp  *    documentation and/or other materials provided with the distribution.
156dbacee2SPoul-Henning Kamp  *
166dbacee2SPoul-Henning Kamp  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
176dbacee2SPoul-Henning Kamp  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
186dbacee2SPoul-Henning Kamp  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
196dbacee2SPoul-Henning Kamp  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
206dbacee2SPoul-Henning Kamp  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
216dbacee2SPoul-Henning Kamp  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
226dbacee2SPoul-Henning Kamp  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
236dbacee2SPoul-Henning Kamp  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
246dbacee2SPoul-Henning Kamp  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
256dbacee2SPoul-Henning Kamp  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
266dbacee2SPoul-Henning Kamp  * SUCH DAMAGE.
276dbacee2SPoul-Henning Kamp  */
286dbacee2SPoul-Henning Kamp 
296dbacee2SPoul-Henning Kamp #include <namespace.h>
306dbacee2SPoul-Henning Kamp #include <stdio.h>
316dbacee2SPoul-Henning Kamp #include <string.h>
326dbacee2SPoul-Henning Kamp #include <stdlib.h>
336dbacee2SPoul-Henning Kamp #include <ctype.h>
346dbacee2SPoul-Henning Kamp #include <wchar.h>
356dbacee2SPoul-Henning Kamp #include <vis.h>
366dbacee2SPoul-Henning Kamp #include <assert.h>
376dbacee2SPoul-Henning Kamp #include <sys/time.h>
386dbacee2SPoul-Henning Kamp #include "printf.h"
396dbacee2SPoul-Henning Kamp 
406dbacee2SPoul-Henning Kamp int
__printf_arginfo_quote(const struct printf_info * pi __unused,size_t n,int * argt)416dbacee2SPoul-Henning Kamp __printf_arginfo_quote(const struct printf_info *pi __unused, size_t n, int *argt)
426dbacee2SPoul-Henning Kamp {
436dbacee2SPoul-Henning Kamp 
446dbacee2SPoul-Henning Kamp 	assert(n >= 1);
456dbacee2SPoul-Henning Kamp 	argt[0] = PA_POINTER;
466dbacee2SPoul-Henning Kamp 	return (1);
476dbacee2SPoul-Henning Kamp }
486dbacee2SPoul-Henning Kamp 
496dbacee2SPoul-Henning Kamp int
__printf_render_quote(struct __printf_io * io,const struct printf_info * pi __unused,const void * const * arg)506dbacee2SPoul-Henning Kamp __printf_render_quote(struct __printf_io *io, const struct printf_info *pi __unused, const void *const *arg)
516dbacee2SPoul-Henning Kamp {
526dbacee2SPoul-Henning Kamp 	const char *str, *p, *t, *o;
53981332f1SPoul-Henning Kamp 	char r[5];
546dbacee2SPoul-Henning Kamp 	int i, ret;
556dbacee2SPoul-Henning Kamp 
566dbacee2SPoul-Henning Kamp 	str = *((const char *const *)arg[0]);
576dbacee2SPoul-Henning Kamp 	if (str == NULL)
586dbacee2SPoul-Henning Kamp 		return (__printf_out(io, pi, "\"(null)\"", 8));
596dbacee2SPoul-Henning Kamp 	if (*str == '\0')
60981332f1SPoul-Henning Kamp 		return (__printf_out(io, pi, "\"\"", 2));
61981332f1SPoul-Henning Kamp 
62981332f1SPoul-Henning Kamp 	for (i = 0, p = str; *p; p++)
636dbacee2SPoul-Henning Kamp 		if (isspace(*p) || *p == '\\' || *p == '"')
646dbacee2SPoul-Henning Kamp 			i++;
656dbacee2SPoul-Henning Kamp 	if (!i)
666dbacee2SPoul-Henning Kamp 		return (__printf_out(io, pi, str, strlen(str)));
676dbacee2SPoul-Henning Kamp 
686dbacee2SPoul-Henning Kamp 	ret = __printf_out(io, pi, "\"", 1);
69981332f1SPoul-Henning Kamp 	for (t = p = str; *p; p++) {
706dbacee2SPoul-Henning Kamp 		o = NULL;
716dbacee2SPoul-Henning Kamp 		if (*p == '\\')
726dbacee2SPoul-Henning Kamp 			o = "\\\\";
736dbacee2SPoul-Henning Kamp 		else if (*p == '\n')
746dbacee2SPoul-Henning Kamp 			o = "\\n";
756dbacee2SPoul-Henning Kamp 		else if (*p == '\r')
766dbacee2SPoul-Henning Kamp 			o = "\\r";
776dbacee2SPoul-Henning Kamp 		else if (*p == '\t')
786dbacee2SPoul-Henning Kamp 			o = "\\t";
796dbacee2SPoul-Henning Kamp 		else if (*p == ' ')
806dbacee2SPoul-Henning Kamp 			o = " ";
816dbacee2SPoul-Henning Kamp 		else if (*p == '"')
826dbacee2SPoul-Henning Kamp 			o = "\\\"";
836dbacee2SPoul-Henning Kamp 		else if (isspace(*p)) {
846dbacee2SPoul-Henning Kamp 			sprintf(r, "\\%03o", *p);
856dbacee2SPoul-Henning Kamp 			o = r;
866dbacee2SPoul-Henning Kamp 		} else
876dbacee2SPoul-Henning Kamp 			continue;
88981332f1SPoul-Henning Kamp 		if (p != t)
896dbacee2SPoul-Henning Kamp 			ret += __printf_out(io, pi, t, p - t);
906dbacee2SPoul-Henning Kamp 		ret += __printf_out(io, pi, o, strlen(o));
91981332f1SPoul-Henning Kamp 		t = p + 1;
926dbacee2SPoul-Henning Kamp 	}
936dbacee2SPoul-Henning Kamp 	if (p != t)
946dbacee2SPoul-Henning Kamp 		ret += __printf_out(io, pi, t, p - t);
956dbacee2SPoul-Henning Kamp 	ret += __printf_out(io, pi, "\"", 1);
966dbacee2SPoul-Henning Kamp 	__printf_flush(io);
976dbacee2SPoul-Henning Kamp 	return(ret);
986dbacee2SPoul-Henning Kamp }
99