annotate mcabber/libjabber/snprintf.c @ 1509:189ffdd944b4

Add a small Coding Style document
author Mikael Berthe <mikael@lilotux.net>
date Sun, 31 Aug 2008 15:21:49 +0200
parents c3ae9251c197
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
25
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
1 /* ====================================================================
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
2 * Copyright (c) 1995-1998 The Apache Group. All rights reserved.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
3 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
4 * Redistribution and use in source and binary forms, with or without
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
5 * modification, are permitted provided that the following conditions
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
6 * are met:
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
7 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
8 * 1. Redistributions of source code must retain the above copyright
414
ec86d759ed54 Trailing whitespace cleanup
Mikael Berthe <mikael@lilotux.net>
parents: 25
diff changeset
9 * notice, this list of conditions and the following disclaimer.
25
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
10 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
11 * 2. Redistributions in binary form must reproduce the above copyright
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
12 * notice, this list of conditions and the following disclaimer in
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
13 * the documentation and/or other materials provided with the
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
14 * distribution.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
15 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
16 * 3. All advertising materials mentioning features or use of this
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
17 * software must display the following acknowledgment:
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
18 * "This product includes software developed by the Apache Group
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
19 * for use in the Apache HTTP server project (http://www.apache.org/)."
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
20 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
21 * 4. The names "Apache Server" and "Apache Group" must not be used to
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
22 * endorse or promote products derived from this software without
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
23 * prior written permission.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
24 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
25 * 5. Redistributions of any form whatsoever must retain the following
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
26 * acknowledgment:
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
27 * "This product includes software developed by the Apache Group
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
28 * for use in the Apache HTTP server project (http://www.apache.org/)."
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
29 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
30 * THIS SOFTWARE IS PROVIDED BY THE APACHE GROUP ``AS IS'' AND ANY
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
31 * EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
32 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
33 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE APACHE GROUP OR
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
34 * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
35 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
36 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
37 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
38 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
39 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
40 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
41 * OF THE POSSIBILITY OF SUCH DAMAGE.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
42 * ====================================================================
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
43 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
44 * This software consists of voluntary contributions made by many
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
45 * individuals on behalf of the Apache Group and was originally based
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
46 * on public domain software written at the National Center for
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
47 * Supercomputing Applications, University of Illinois, Urbana-Champaign.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
48 * For more information on the Apache Group and the Apache HTTP server
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
49 * project, please see <http://www.apache.org/>.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
50 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
51 * This code is based on, and used with the permission of, the
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
52 * SIO stdio-replacement strx_* functions by Panos Tsirigotis
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
53 * <panos@alumni.cs.colorado.edu> for xinetd.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
54 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
55
417
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
56 /**
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
57 * @file snprintf.c
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
58 * @brief implement snprintf if not present in the libc
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
59 *
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
60 * snprintf is not implemented by all libc implementations, this file implements this
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
61 * function, if it is not already present. You should not call any of the functions
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
62 * in this file directly!
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
63 */
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
64
25
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
65 #include <libxode.h>
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
66
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
67 #if !defined(HAVE_SNPRINTF) || !defined(HAVE_VSNPRINTF)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
68
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
69 #include <stdio.h>
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
70 #include <ctype.h>
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
71 #include <sys/types.h>
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
72 #include <stdarg.h>
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
73 #include <string.h>
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
74 #include <stdlib.h>
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
75 #include <math.h>
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
76
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
77
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
78 #ifdef HAVE_GCVT
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
79
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
80 #define ap_ecvt ecvt
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
81 #define ap_fcvt fcvt
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
82 #define ap_gcvt gcvt
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
83
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
84 #else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
85
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
86 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
87 * cvt.c - IEEE floating point formatting routines for FreeBSD
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
88 * from GNU libc-4.6.27
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
89 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
90
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
91 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
92 * ap_ecvt converts to decimal
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
93 * the number of digits is specified by ndigit
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
94 * decpt is set to the position of the decimal point
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
95 * sign is set to 0 for positive, 1 for negative
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
96 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
97
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
98 #define NDIG 80
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
99
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
100 static char *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
101 ap_cvt(double arg, int ndigits, int *decpt, int *sign, int eflag)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
102 {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
103 register int r2;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
104 double fi, fj;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
105 register char *p, *p1;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
106 static char buf[NDIG];
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
107
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
108 if (ndigits >= NDIG - 1)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
109 ndigits = NDIG - 2;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
110 r2 = 0;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
111 *sign = 0;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
112 p = &buf[0];
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
113 if (arg < 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
114 *sign = 1;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
115 arg = -arg;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
116 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
117 arg = modf(arg, &fi);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
118 p1 = &buf[NDIG];
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
119 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
120 * Do integer part
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
121 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
122 if (fi != 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
123 p1 = &buf[NDIG];
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
124 while (fi != 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
125 fj = modf(fi / 10, &fi);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
126 *--p1 = (int) ((fj + .03) * 10) + '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
127 r2++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
128 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
129 while (p1 < &buf[NDIG])
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
130 *p++ = *p1++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
131 } else if (arg > 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
132 while ((fj = arg * 10) < 1) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
133 arg = fj;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
134 r2--;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
135 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
136 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
137 p1 = &buf[ndigits];
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
138 if (eflag == 0)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
139 p1 += r2;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
140 *decpt = r2;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
141 if (p1 < &buf[0]) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
142 buf[0] = '\0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
143 return (buf);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
144 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
145 while (p <= p1 && p < &buf[NDIG]) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
146 arg *= 10;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
147 arg = modf(arg, &fj);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
148 *p++ = (int) fj + '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
149 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
150 if (p1 >= &buf[NDIG]) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
151 buf[NDIG - 1] = '\0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
152 return (buf);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
153 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
154 p = p1;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
155 *p1 += 5;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
156 while (*p1 > '9') {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
157 *p1 = '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
158 if (p1 > buf)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
159 ++ * --p1;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
160 else {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
161 *p1 = '1';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
162 (*decpt)++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
163 if (eflag == 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
164 if (p > buf)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
165 *p = '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
166 p++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
167 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
168 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
169 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
170 *p = '\0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
171 return (buf);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
172 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
173
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
174 static char *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
175 ap_ecvt(double arg, int ndigits, int *decpt, int *sign)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
176 {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
177 return (ap_cvt(arg, ndigits, decpt, sign, 1));
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
178 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
179
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
180 static char *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
181 ap_fcvt(double arg, int ndigits, int *decpt, int *sign)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
182 {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
183 return (ap_cvt(arg, ndigits, decpt, sign, 0));
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
184 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
185
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
186 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
187 * ap_gcvt - Floating output conversion to
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
188 * minimal length string
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
189 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
190
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
191 static char *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
192 ap_gcvt(double number, int ndigit, char *buf)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
193 {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
194 int sign, decpt;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
195 register char *p1, *p2;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
196 int i;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
197
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
198 p1 = ap_ecvt(number, ndigit, &decpt, &sign);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
199 p2 = buf;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
200 if (sign)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
201 *p2++ = '-';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
202 for (i = ndigit - 1; i > 0 && p1[i] == '0'; i--)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
203 ndigit--;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
204 if ((decpt >= 0 && decpt - ndigit > 4)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
205 || (decpt < 0 && decpt < -3)) { /* use E-style */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
206 decpt--;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
207 *p2++ = *p1++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
208 *p2++ = '.';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
209 for (i = 1; i < ndigit; i++)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
210 *p2++ = *p1++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
211 *p2++ = 'e';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
212 if (decpt < 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
213 decpt = -decpt;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
214 *p2++ = '-';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
215 } else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
216 *p2++ = '+';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
217 if (decpt / 100 > 0)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
218 *p2++ = decpt / 100 + '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
219 if (decpt / 10 > 0)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
220 *p2++ = (decpt % 100) / 10 + '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
221 *p2++ = decpt % 10 + '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
222 } else {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
223 if (decpt <= 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
224 if (*p1 != '0')
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
225 *p2++ = '.';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
226 while (decpt < 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
227 decpt++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
228 *p2++ = '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
229 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
230 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
231 for (i = 1; i <= ndigit; i++) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
232 *p2++ = *p1++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
233 if (i == decpt)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
234 *p2++ = '.';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
235 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
236 if (ndigit < decpt) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
237 while (ndigit++ < decpt)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
238 *p2++ = '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
239 *p2++ = '.';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
240 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
241 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
242 if (p2[-1] == '.')
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
243 p2--;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
244 *p2 = '\0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
245 return (buf);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
246 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
247
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
248 #endif /* HAVE_CVT */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
249
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
250 typedef enum {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
251 NO = 0, YES = 1
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
252 } boolean_e;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
253
417
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
254 #ifndef FALSE
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
255 # define FALSE 0
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
256 #endif
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
257 #ifndef TRUE
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
258 # define TRUE 1
c3ae9251c197 Sync libjabber with upstream
Mikael Berthe <mikael@lilotux.net>
parents: 414
diff changeset
259 #endif
25
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
260 #define NUL '\0'
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
261 #define INT_NULL ((int *)0)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
262 #define WIDE_INT long
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
263
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
264 typedef WIDE_INT wide_int;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
265 typedef unsigned WIDE_INT u_wide_int;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
266 typedef int bool_int;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
267
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
268 #define S_NULL "(null)"
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
269 #define S_NULL_LEN 6
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
270
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
271 #define FLOAT_DIGITS 6
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
272 #define EXPONENT_LENGTH 10
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
273
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
274 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
275 * NUM_BUF_SIZE is the size of the buffer used for arithmetic conversions
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
276 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
277 * XXX: this is a magic number; do not decrease it
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
278 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
279 #define NUM_BUF_SIZE 512
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
280
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
281
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
282 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
283 * Descriptor for buffer area
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
284 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
285 struct buf_area {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
286 char *buf_end;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
287 char *nextb; /* pointer to next byte to read/write */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
288 };
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
289
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
290 typedef struct buf_area buffy;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
291
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
292 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
293 * The INS_CHAR macro inserts a character in the buffer and writes
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
294 * the buffer back to disk if necessary
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
295 * It uses the char pointers sp and bep:
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
296 * sp points to the next available character in the buffer
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
297 * bep points to the end-of-buffer+1
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
298 * While using this macro, note that the nextb pointer is NOT updated.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
299 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
300 * NOTE: Evaluation of the c argument should not have any side-effects
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
301 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
302 #define INS_CHAR( c, sp, bep, cc ) \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
303 { \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
304 if ( sp < bep ) \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
305 { \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
306 *sp++ = c ; \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
307 cc++ ; \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
308 } \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
309 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
310
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
311 #define NUM( c ) ( c - '0' )
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
312
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
313 #define STR_TO_DEC( str, num ) \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
314 num = NUM( *str++ ) ; \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
315 while ( isdigit((int)*str ) ) \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
316 { \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
317 num *= 10 ; \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
318 num += NUM( *str++ ) ; \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
319 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
320
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
321 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
322 * This macro does zero padding so that the precision
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
323 * requirement is satisfied. The padding is done by
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
324 * adding '0's to the left of the string that is going
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
325 * to be printed.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
326 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
327 #define FIX_PRECISION( adjust, precision, s, s_len ) \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
328 if ( adjust ) \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
329 while ( s_len < precision ) \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
330 { \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
331 *--s = '0' ; \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
332 s_len++ ; \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
333 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
334
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
335 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
336 * Macro that does padding. The padding is done by printing
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
337 * the character ch.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
338 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
339 #define PAD( width, len, ch ) do \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
340 { \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
341 INS_CHAR( ch, sp, bep, cc ) ; \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
342 width-- ; \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
343 } \
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
344 while ( width > len )
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
345
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
346 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
347 * Prefix the character ch to the string str
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
348 * Increase length
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
349 * Set the has_prefix flag
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
350 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
351 #define PREFIX( str, length, ch ) *--str = ch ; length++ ; has_prefix = YES
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
352
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
353
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
354 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
355 * Convert num to its decimal format.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
356 * Return value:
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
357 * - a pointer to a string containing the number (no sign)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
358 * - len contains the length of the string
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
359 * - is_negative is set to TRUE or FALSE depending on the sign
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
360 * of the number (always set to FALSE if is_unsigned is TRUE)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
361 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
362 * The caller provides a buffer for the string: that is the buf_end argument
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
363 * which is a pointer to the END of the buffer + 1 (i.e. if the buffer
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
364 * is declared as buf[ 100 ], buf_end should be &buf[ 100 ])
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
365 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
366 static char *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
367 conv_10(register wide_int num, register bool_int is_unsigned,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
368 register bool_int * is_negative, char *buf_end, register int *len)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
369 {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
370 register char *p = buf_end;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
371 register u_wide_int magnitude;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
372
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
373 if (is_unsigned) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
374 magnitude = (u_wide_int) num;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
375 *is_negative = FALSE;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
376 } else {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
377 *is_negative = (num < 0);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
378
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
379 /*
414
ec86d759ed54 Trailing whitespace cleanup
Mikael Berthe <mikael@lilotux.net>
parents: 25
diff changeset
380 * On a 2's complement machine, negating the most negative integer
25
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
381 * results in a number that cannot be represented as a signed integer.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
382 * Here is what we do to obtain the number's magnitude:
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
383 * a. add 1 to the number
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
384 * b. negate it (becomes positive)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
385 * c. convert it to unsigned
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
386 * d. add 1
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
387 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
388 if (*is_negative) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
389 wide_int t = num + 1;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
390
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
391 magnitude = ((u_wide_int) - t) + 1;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
392 } else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
393 magnitude = (u_wide_int) num;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
394 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
395
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
396 /*
414
ec86d759ed54 Trailing whitespace cleanup
Mikael Berthe <mikael@lilotux.net>
parents: 25
diff changeset
397 * We use a do-while loop so that we write at least 1 digit
25
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
398 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
399 do {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
400 register u_wide_int new_magnitude = magnitude / 10;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
401
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
402 *--p = magnitude - new_magnitude * 10 + '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
403 magnitude = new_magnitude;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
404 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
405 while (magnitude);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
406
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
407 *len = buf_end - p;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
408 return (p);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
409 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
410
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
411
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
412
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
413 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
414 * Convert a floating point number to a string formats 'f', 'e' or 'E'.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
415 * The result is placed in buf, and len denotes the length of the string
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
416 * The sign is returned in the is_negative argument (and is not placed
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
417 * in buf).
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
418 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
419 static char *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
420 conv_fp(register char format, register double num,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
421 boolean_e add_dp, int precision, bool_int * is_negative, char *buf, int *len)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
422 {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
423 register char *s = buf;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
424 register char *p;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
425 int decimal_point;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
426
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
427 if (format == 'f')
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
428 p = ap_fcvt(num, precision, &decimal_point, is_negative);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
429 else /* either e or E format */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
430 p = ap_ecvt(num, precision + 1, &decimal_point, is_negative);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
431
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
432 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
433 * Check for Infinity and NaN
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
434 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
435 if (isalpha((int)*p)) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
436 *len = strlen(strcpy(buf, p));
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
437 *is_negative = FALSE;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
438 return (buf);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
439 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
440 if (format == 'f') {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
441 if (decimal_point <= 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
442 *s++ = '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
443 if (precision > 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
444 *s++ = '.';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
445 while (decimal_point++ < 0)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
446 *s++ = '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
447 } else if (add_dp) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
448 *s++ = '.';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
449 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
450 } else {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
451 while (decimal_point-- > 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
452 *s++ = *p++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
453 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
454 if (precision > 0 || add_dp) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
455 *s++ = '.';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
456 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
457 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
458 } else {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
459 *s++ = *p++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
460 if (precision > 0 || add_dp)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
461 *s++ = '.';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
462 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
463
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
464 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
465 * copy the rest of p, the NUL is NOT copied
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
466 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
467 while (*p)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
468 *s++ = *p++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
469
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
470 if (format != 'f') {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
471 char temp[EXPONENT_LENGTH]; /* for exponent conversion */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
472 int t_len;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
473 bool_int exponent_is_negative;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
474
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
475 *s++ = format; /* either e or E */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
476 decimal_point--;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
477 if (decimal_point != 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
478 p = conv_10((wide_int) decimal_point, FALSE, &exponent_is_negative,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
479 &temp[EXPONENT_LENGTH], &t_len);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
480 *s++ = exponent_is_negative ? '-' : '+';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
481
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
482 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
483 * Make sure the exponent has at least 2 digits
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
484 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
485 if (t_len == 1)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
486 *s++ = '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
487 while (t_len--)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
488 *s++ = *p++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
489 } else {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
490 *s++ = '+';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
491 *s++ = '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
492 *s++ = '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
493 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
494 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
495 *len = s - buf;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
496 return (buf);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
497 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
498
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
499
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
500 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
501 * Convert num to a base X number where X is a power of 2. nbits determines X.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
502 * For example, if nbits is 3, we do base 8 conversion
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
503 * Return value:
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
504 * a pointer to a string containing the number
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
505 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
506 * The caller provides a buffer for the string: that is the buf_end argument
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
507 * which is a pointer to the END of the buffer + 1 (i.e. if the buffer
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
508 * is declared as buf[ 100 ], buf_end should be &buf[ 100 ])
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
509 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
510 static char *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
511 conv_p2(register u_wide_int num, register int nbits,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
512 char format, char *buf_end, register int *len)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
513 {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
514 register int mask = (1 << nbits) - 1;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
515 register char *p = buf_end;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
516 static char low_digits[] = "0123456789abcdef";
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
517 static char upper_digits[] = "0123456789ABCDEF";
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
518 register char *digits = (format == 'X') ? upper_digits : low_digits;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
519
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
520 do {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
521 *--p = digits[num & mask];
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
522 num >>= nbits;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
523 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
524 while (num);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
525
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
526 *len = buf_end - p;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
527 return (p);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
528 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
529
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
530
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
531 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
532 * Do format conversion placing the output in buffer
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
533 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
534 static int format_converter(register buffy * odp, const char *fmt,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
535 va_list ap)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
536 {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
537 register char *sp;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
538 register char *bep;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
539 register int cc = 0;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
540 register int i;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
541
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
542 register char *s = NULL;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
543 char *q;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
544 int s_len;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
545
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
546 register int min_width = 0;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
547 int precision = 0;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
548 enum {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
549 LEFT, RIGHT
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
550 } adjust;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
551 char pad_char;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
552 char prefix_char;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
553
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
554 double fp_num;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
555 wide_int i_num = (wide_int) 0;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
556 u_wide_int ui_num;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
557
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
558 char num_buf[NUM_BUF_SIZE];
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
559 char char_buf[2]; /* for printing %% and %<unknown> */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
560
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
561 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
562 * Flag variables
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
563 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
564 boolean_e is_long;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
565 boolean_e alternate_form;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
566 boolean_e print_sign;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
567 boolean_e print_blank;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
568 boolean_e adjust_precision;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
569 boolean_e adjust_width;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
570 bool_int is_negative;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
571
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
572 sp = odp->nextb;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
573 bep = odp->buf_end;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
574
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
575 while (*fmt) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
576 if (*fmt != '%') {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
577 INS_CHAR(*fmt, sp, bep, cc);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
578 } else {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
579 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
580 * Default variable settings
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
581 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
582 adjust = RIGHT;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
583 alternate_form = print_sign = print_blank = NO;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
584 pad_char = ' ';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
585 prefix_char = NUL;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
586
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
587 fmt++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
588
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
589 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
590 * Try to avoid checking for flags, width or precision
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
591 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
592 if (isascii((int)*fmt) && !islower((int)*fmt)) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
593 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
594 * Recognize flags: -, #, BLANK, +
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
595 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
596 for (;; fmt++) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
597 if (*fmt == '-')
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
598 adjust = LEFT;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
599 else if (*fmt == '+')
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
600 print_sign = YES;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
601 else if (*fmt == '#')
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
602 alternate_form = YES;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
603 else if (*fmt == ' ')
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
604 print_blank = YES;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
605 else if (*fmt == '0')
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
606 pad_char = '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
607 else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
608 break;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
609 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
610
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
611 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
612 * Check if a width was specified
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
613 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
614 if (isdigit((int)*fmt)) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
615 STR_TO_DEC(fmt, min_width);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
616 adjust_width = YES;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
617 } else if (*fmt == '*') {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
618 min_width = va_arg(ap, int);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
619 fmt++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
620 adjust_width = YES;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
621 if (min_width < 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
622 adjust = LEFT;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
623 min_width = -min_width;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
624 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
625 } else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
626 adjust_width = NO;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
627
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
628 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
629 * Check if a precision was specified
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
630 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
631 * XXX: an unreasonable amount of precision may be specified
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
632 * resulting in overflow of num_buf. Currently we
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
633 * ignore this possibility.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
634 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
635 if (*fmt == '.') {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
636 adjust_precision = YES;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
637 fmt++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
638 if (isdigit((int)*fmt)) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
639 STR_TO_DEC(fmt, precision);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
640 } else if (*fmt == '*') {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
641 precision = va_arg(ap, int);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
642 fmt++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
643 if (precision < 0)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
644 precision = 0;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
645 } else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
646 precision = 0;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
647 } else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
648 adjust_precision = NO;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
649 } else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
650 adjust_precision = adjust_width = NO;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
651
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
652 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
653 * Modifier check
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
654 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
655 if (*fmt == 'l') {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
656 is_long = YES;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
657 fmt++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
658 } else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
659 is_long = NO;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
660
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
661 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
662 * Argument extraction and printing.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
663 * First we determine the argument type.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
664 * Then, we convert the argument to a string.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
665 * On exit from the switch, s points to the string that
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
666 * must be printed, s_len has the length of the string
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
667 * The precision requirements, if any, are reflected in s_len.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
668 *
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
669 * NOTE: pad_char may be set to '0' because of the 0 flag.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
670 * It is reset to ' ' by non-numeric formats
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
671 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
672 switch (*fmt) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
673 case 'u':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
674 if (is_long)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
675 i_num = va_arg(ap, u_wide_int);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
676 else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
677 i_num = (wide_int) va_arg(ap, unsigned int);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
678 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
679 * The rest also applies to other integer formats, so fall
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
680 * into that case.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
681 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
682 case 'd':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
683 case 'i':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
684 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
685 * Get the arg if we haven't already.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
686 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
687 if ((*fmt) != 'u') {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
688 if (is_long)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
689 i_num = va_arg(ap, wide_int);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
690 else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
691 i_num = (wide_int) va_arg(ap, int);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
692 };
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
693 s = conv_10(i_num, (*fmt) == 'u', &is_negative,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
694 &num_buf[NUM_BUF_SIZE], &s_len);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
695 FIX_PRECISION(adjust_precision, precision, s, s_len);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
696
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
697 if (*fmt != 'u') {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
698 if (is_negative)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
699 prefix_char = '-';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
700 else if (print_sign)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
701 prefix_char = '+';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
702 else if (print_blank)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
703 prefix_char = ' ';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
704 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
705 break;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
706
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
707
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
708 case 'o':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
709 if (is_long)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
710 ui_num = va_arg(ap, u_wide_int);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
711 else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
712 ui_num = (u_wide_int) va_arg(ap, unsigned int);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
713 s = conv_p2(ui_num, 3, *fmt,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
714 &num_buf[NUM_BUF_SIZE], &s_len);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
715 FIX_PRECISION(adjust_precision, precision, s, s_len);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
716 if (alternate_form && *s != '0') {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
717 *--s = '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
718 s_len++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
719 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
720 break;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
721
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
722
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
723 case 'x':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
724 case 'X':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
725 if (is_long)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
726 ui_num = (u_wide_int) va_arg(ap, u_wide_int);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
727 else
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
728 ui_num = (u_wide_int) va_arg(ap, unsigned int);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
729 s = conv_p2(ui_num, 4, *fmt,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
730 &num_buf[NUM_BUF_SIZE], &s_len);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
731 FIX_PRECISION(adjust_precision, precision, s, s_len);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
732 if (alternate_form && i_num != 0) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
733 *--s = *fmt; /* 'x' or 'X' */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
734 *--s = '0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
735 s_len += 2;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
736 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
737 break;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
738
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
739
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
740 case 's':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
741 s = va_arg(ap, char *);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
742 if (s != NULL) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
743 s_len = strlen(s);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
744 if (adjust_precision && precision < s_len)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
745 s_len = precision;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
746 } else {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
747 s = S_NULL;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
748 s_len = S_NULL_LEN;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
749 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
750 pad_char = ' ';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
751 break;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
752
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
753
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
754 case 'f':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
755 case 'e':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
756 case 'E':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
757 fp_num = va_arg(ap, double);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
758
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
759 s = conv_fp(*fmt, fp_num, alternate_form,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
760 (adjust_precision == NO) ? FLOAT_DIGITS : precision,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
761 &is_negative, &num_buf[1], &s_len);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
762 if (is_negative)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
763 prefix_char = '-';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
764 else if (print_sign)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
765 prefix_char = '+';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
766 else if (print_blank)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
767 prefix_char = ' ';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
768 break;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
769
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
770
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
771 case 'g':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
772 case 'G':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
773 if (adjust_precision == NO)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
774 precision = FLOAT_DIGITS;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
775 else if (precision == 0)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
776 precision = 1;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
777 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
778 * * We use &num_buf[ 1 ], so that we have room for the sign
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
779 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
780 s = ap_gcvt(va_arg(ap, double), precision, &num_buf[1]);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
781 if (*s == '-')
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
782 prefix_char = *s++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
783 else if (print_sign)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
784 prefix_char = '+';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
785 else if (print_blank)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
786 prefix_char = ' ';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
787
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
788 s_len = strlen(s);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
789
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
790 if (alternate_form && (q = strchr(s, '.')) == NULL)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
791 s[s_len++] = '.';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
792 if (*fmt == 'G' && (q = strchr(s, 'e')) != NULL)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
793 *q = 'E';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
794 break;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
795
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
796
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
797 case 'c':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
798 char_buf[0] = (char) (va_arg(ap, int));
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
799 s = &char_buf[0];
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
800 s_len = 1;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
801 pad_char = ' ';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
802 break;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
803
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
804
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
805 case '%':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
806 char_buf[0] = '%';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
807 s = &char_buf[0];
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
808 s_len = 1;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
809 pad_char = ' ';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
810 break;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
811
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
812
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
813 case 'n':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
814 *(va_arg(ap, int *)) = cc;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
815 break;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
816
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
817 /*
414
ec86d759ed54 Trailing whitespace cleanup
Mikael Berthe <mikael@lilotux.net>
parents: 25
diff changeset
818 * Always extract the argument as a "char *" pointer. We
ec86d759ed54 Trailing whitespace cleanup
Mikael Berthe <mikael@lilotux.net>
parents: 25
diff changeset
819 * should be using "void *" but there are still machines
25
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
820 * that don't understand it.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
821 * If the pointer size is equal to the size of an unsigned
414
ec86d759ed54 Trailing whitespace cleanup
Mikael Berthe <mikael@lilotux.net>
parents: 25
diff changeset
822 * integer we convert the pointer to a hex number, otherwise
25
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
823 * we print "%p" to indicate that we don't handle "%p".
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
824 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
825 case 'p':
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
826 ui_num = (u_wide_int) va_arg(ap, char *);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
827
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
828 if (sizeof(char *) <= sizeof(u_wide_int))
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
829 s = conv_p2(ui_num, 4, 'x',
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
830 &num_buf[NUM_BUF_SIZE], &s_len);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
831 else {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
832 s = "%p";
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
833 s_len = 2;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
834 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
835 pad_char = ' ';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
836 break;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
837
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
838
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
839 case NUL:
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
840 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
841 * The last character of the format string was %.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
842 * We ignore it.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
843 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
844 continue;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
845
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
846
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
847 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
848 * The default case is for unrecognized %'s.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
849 * We print %<char> to help the user identify what
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
850 * option is not understood.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
851 * This is also useful in case the user wants to pass
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
852 * the output of format_converter to another function
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
853 * that understands some other %<char> (like syslog).
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
854 * Note that we can't point s inside fmt because the
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
855 * unknown <char> could be preceded by width etc.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
856 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
857 default:
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
858 char_buf[0] = '%';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
859 char_buf[1] = *fmt;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
860 s = char_buf;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
861 s_len = 2;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
862 pad_char = ' ';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
863 break;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
864 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
865
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
866 if (prefix_char != NUL) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
867 *--s = prefix_char;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
868 s_len++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
869 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
870 if (adjust_width && adjust == RIGHT && min_width > s_len) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
871 if (pad_char == '0' && prefix_char != NUL) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
872 INS_CHAR(*s, sp, bep, cc)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
873 s++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
874 s_len--;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
875 min_width--;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
876 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
877 PAD(min_width, s_len, pad_char);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
878 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
879 /*
414
ec86d759ed54 Trailing whitespace cleanup
Mikael Berthe <mikael@lilotux.net>
parents: 25
diff changeset
880 * Print the string s.
25
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
881 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
882 for (i = s_len; i != 0; i--) {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
883 INS_CHAR(*s, sp, bep, cc);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
884 s++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
885 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
886
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
887 if (adjust_width && adjust == LEFT && min_width > s_len)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
888 PAD(min_width, s_len, pad_char);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
889 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
890 fmt++;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
891 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
892 odp->nextb = sp;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
893 return (cc);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
894 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
895
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
896
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
897 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
898 * This is the general purpose conversion function.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
899 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
900 static void strx_printv(int *ccp, char *buf, size_t len, const char *format,
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
901 va_list ap)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
902 {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
903 buffy od;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
904 int cc;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
905
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
906 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
907 * First initialize the descriptor
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
908 * Notice that if no length is given, we initialize buf_end to the
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
909 * highest possible address.
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
910 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
911 od.buf_end = len ? &buf[len] : (char *) ~0;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
912 od.nextb = buf;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
913
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
914 /*
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
915 * Do the conversion
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
916 */
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
917 cc = format_converter(&od, format, ap);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
918 if (len == 0 || od.nextb <= od.buf_end)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
919 *(od.nextb) = '\0';
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
920 if (ccp)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
921 *ccp = cc;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
922 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
923
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
924
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
925 int ap_snprintf(char *buf, size_t len, const char *format,...)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
926 {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
927 int cc;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
928 va_list ap;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
929
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
930 va_start(ap, format);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
931 strx_printv(&cc, buf, (len - 1), format, ap);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
932 va_end(ap);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
933 return (cc);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
934 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
935
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
936
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
937 int ap_vsnprintf(char *buf, size_t len, const char *format, va_list ap)
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
938 {
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
939 int cc;
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
940
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
941 strx_printv(&cc, buf, (len - 1), format, ap);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
942 return (cc);
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
943 }
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
944
bf3d6e241714 [/trunk] Changeset 41 by mikael
mikael
parents:
diff changeset
945 #endif /* HAVE_SNPRINTF */