| Juan Cespedes | d44c6b8 | 1998-09-25 14:48:42 +0200 | [diff] [blame] | 1 | #if HAVE_CONFIG_H |
| 2 | #include "config.h" |
| 3 | #endif |
| 4 | |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 5 | #include <ctype.h> |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 6 | #include <stdio.h> |
| 7 | #include <stdlib.h> |
| Juan Cespedes | d914a20 | 2004-11-10 00:15:33 +0100 | [diff] [blame] | 8 | #include <string.h> |
| Juan Cespedes | 2c4a8cb | 1998-03-11 23:33:18 +0100 | [diff] [blame] | 9 | #include <limits.h> |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 10 | |
| 11 | #include "ltrace.h" |
| 12 | #include "options.h" |
| 13 | |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 14 | static int display_char(int what); |
| Steve Fink | 6a48a6d | 2006-08-07 04:29:06 +0200 | [diff] [blame] | 15 | static int display_string(enum tof type, struct process *proc, |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 16 | void* addr, size_t maxlen); |
| 17 | static int display_value(enum tof type, struct process *proc, |
| 18 | long value, arg_type_info *info); |
| 19 | static int display_unknown(enum tof type, struct process *proc, long value); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 20 | static int display_format(enum tof type, struct process *proc, int arg_num); |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 21 | |
| Steve Fink | 6a48a6d | 2006-08-07 04:29:06 +0200 | [diff] [blame] | 22 | static int string_maxlength = INT_MAX; |
| 23 | |
| 24 | static long get_length(enum tof type, struct process *proc, int len_spec) |
| 25 | { |
| 26 | if (len_spec > 0) |
| 27 | return len_spec; |
| 28 | return gimme_arg(type, proc, -len_spec - 1); |
| 29 | } |
| 30 | |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 31 | static int display_pointer(enum tof type, struct process *proc, long value, |
| 32 | arg_type_info * info) |
| 33 | { |
| 34 | long pointed_to; |
| 35 | arg_type_info *inner = info->u.ptr_info.info; |
| 36 | |
| 37 | if (value == 0) |
| 38 | return fprintf(output, "NULL"); |
| 39 | else if (umovelong(proc, (void *) value, &pointed_to) < 0) |
| 40 | return fprintf(output, "?"); |
| 41 | else |
| 42 | return display_value(type, proc, pointed_to, inner); |
| 43 | } |
| 44 | |
| 45 | /* Args: |
| 46 | type - syscall or shared library function or memory |
| 47 | proc - information about the traced process |
| 48 | value - the value to display |
| 49 | info - the description of the type to display |
| 50 | */ |
| 51 | int display_value(enum tof type, struct process *proc, |
| 52 | long value, arg_type_info *info) |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 53 | { |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 54 | int tmp; |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 55 | |
| Steve Fink | 6a48a6d | 2006-08-07 04:29:06 +0200 | [diff] [blame] | 56 | switch (info->type) { |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 57 | case ARGTYPE_VOID: |
| 58 | return 0; |
| Steve Fink | 6b17583 | 2006-08-07 04:37:33 +0200 | [diff] [blame] | 59 | case ARGTYPE_IGNORE: |
| 60 | return 0; /* Empty gap between commas */ |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 61 | case ARGTYPE_INT: |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 62 | return fprintf(output, "%d", (int) value); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 63 | case ARGTYPE_UINT: |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 64 | return fprintf(output, "%u", (unsigned) value); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 65 | case ARGTYPE_LONG: |
| 66 | if (proc->mask_32bit) |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 67 | return fprintf(output, "%d", (int) value); |
| 68 | else |
| 69 | return fprintf(output, "%ld", value); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 70 | case ARGTYPE_ULONG: |
| 71 | if (proc->mask_32bit) |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 72 | return fprintf(output, "%u", (unsigned) value); |
| 73 | else |
| 74 | return fprintf(output, "%lu", (unsigned long) value); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 75 | case ARGTYPE_OCTAL: |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 76 | return fprintf(output, "0%o", (unsigned) value); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 77 | case ARGTYPE_CHAR: |
| 78 | tmp = fprintf(output, "'"); |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 79 | tmp += display_char(value == -1 ? value : (char) value); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 80 | tmp += fprintf(output, "'"); |
| 81 | return tmp; |
| 82 | case ARGTYPE_ADDR: |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 83 | if (!value) |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 84 | return fprintf(output, "NULL"); |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 85 | else |
| 86 | return fprintf(output, "0x%08lx", value); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 87 | case ARGTYPE_FORMAT: |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 88 | fprintf(stderr, "Should never encounter a format anywhere but at the top level (for now?)\n"); |
| 89 | exit(1); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 90 | case ARGTYPE_STRING: |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 91 | return display_string(type, proc, (void*) value, |
| Steve Fink | 6a48a6d | 2006-08-07 04:29:06 +0200 | [diff] [blame] | 92 | string_maxlength); |
| 93 | case ARGTYPE_STRING_N: |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 94 | return display_string(type, proc, (void*) value, |
| Steve Fink | 6a48a6d | 2006-08-07 04:29:06 +0200 | [diff] [blame] | 95 | get_length(type, proc, |
| 96 | info->u.string_n_info.size_spec)); |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 97 | case ARGTYPE_POINTER: |
| 98 | return display_pointer(type, proc, value, info); |
| 99 | case ARGTYPE_UNKNOWN: |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 100 | default: |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 101 | return display_unknown(type, proc, value); |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 102 | } |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 103 | } |
| 104 | |
| 105 | int display_arg(enum tof type, struct process *proc, int arg_num, |
| 106 | arg_type_info * info) |
| 107 | { |
| 108 | long arg; |
| 109 | |
| 110 | if (info->type == ARGTYPE_VOID) { |
| 111 | return 0; |
| 112 | } else if (info->type == ARGTYPE_FORMAT) { |
| 113 | return display_format(type, proc, arg_num); |
| 114 | } else { |
| 115 | arg = gimme_arg(type, proc, arg_num); |
| 116 | return display_value(type, proc, arg, info); |
| 117 | } |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 118 | } |
| 119 | |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 120 | static int display_char(int what) |
| 121 | { |
| 122 | switch (what) { |
| 123 | case -1: |
| 124 | return fprintf(output, "EOF"); |
| 125 | case '\r': |
| 126 | return fprintf(output, "\\r"); |
| 127 | case '\n': |
| 128 | return fprintf(output, "\\n"); |
| 129 | case '\t': |
| 130 | return fprintf(output, "\\t"); |
| 131 | case '\b': |
| 132 | return fprintf(output, "\\b"); |
| 133 | case '\\': |
| 134 | return fprintf(output, "\\\\"); |
| 135 | default: |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 136 | if (isprint(what)) { |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 137 | return fprintf(output, "%c", what); |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 138 | } else { |
| 139 | return fprintf(output, "\\%03o", (unsigned char)what); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 140 | } |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 141 | } |
| 142 | } |
| 143 | |
| Juan Cespedes | 2c4a8cb | 1998-03-11 23:33:18 +0100 | [diff] [blame] | 144 | #define MIN(a,b) (((a)<(b)) ? (a) : (b)) |
| 145 | |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 146 | static int display_string(enum tof type, struct process *proc, void *addr, |
| 147 | size_t maxlength) |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 148 | { |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 149 | unsigned char *str1; |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 150 | int i; |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 151 | int len = 0; |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 152 | |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 153 | if (!addr) { |
| 154 | return fprintf(output, "NULL"); |
| 155 | } |
| 156 | |
| Steve Fink | 6a48a6d | 2006-08-07 04:29:06 +0200 | [diff] [blame] | 157 | str1 = malloc(MIN(opt_s, maxlength) + 3); |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 158 | if (!str1) { |
| 159 | return fprintf(output, "???"); |
| 160 | } |
| Steve Fink | 6a48a6d | 2006-08-07 04:29:06 +0200 | [diff] [blame] | 161 | umovestr(proc, addr, MIN(opt_s, maxlength) + 1, str1); |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 162 | len = fprintf(output, "\""); |
| Steve Fink | 6a48a6d | 2006-08-07 04:29:06 +0200 | [diff] [blame] | 163 | for (i = 0; i < MIN(opt_s, maxlength); i++) { |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 164 | if (str1[i]) { |
| 165 | len += display_char(str1[i]); |
| 166 | } else { |
| 167 | break; |
| 168 | } |
| 169 | } |
| 170 | len += fprintf(output, "\""); |
| Steve Fink | 6a48a6d | 2006-08-07 04:29:06 +0200 | [diff] [blame] | 171 | if (str1[i] && (opt_s <= maxlength)) { |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 172 | len += fprintf(output, "..."); |
| 173 | } |
| 174 | free(str1); |
| 175 | return len; |
| 176 | } |
| 177 | |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 178 | static int display_unknown(enum tof type, struct process *proc, long value) |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 179 | { |
| Ian Wienand | 9a2ad35 | 2006-02-20 22:44:45 +0100 | [diff] [blame] | 180 | if (proc->mask_32bit) { |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 181 | if ((int)value < 1000000 && (int)value > -1000000) |
| 182 | return fprintf(output, "%d", (int)value); |
| Ian Wienand | 9a2ad35 | 2006-02-20 22:44:45 +0100 | [diff] [blame] | 183 | else |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 184 | return fprintf(output, "%p", (void *)value); |
| 185 | } else if (value < 1000000 && value > -1000000) { |
| 186 | return fprintf(output, "%ld", value); |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 187 | } else { |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 188 | return fprintf(output, "%p", (void *)value); |
| Juan Cespedes | 5e01f65 | 1998-03-08 22:31:44 +0100 | [diff] [blame] | 189 | } |
| 190 | } |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 191 | |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 192 | static int display_format(enum tof type, struct process *proc, int arg_num) |
| 193 | { |
| 194 | void *addr; |
| 195 | unsigned char *str1; |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 196 | int i; |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 197 | int len = 0; |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 198 | |
| 199 | addr = (void *)gimme_arg(type, proc, arg_num); |
| 200 | if (!addr) { |
| 201 | return fprintf(output, "NULL"); |
| 202 | } |
| 203 | |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 204 | str1 = malloc(MIN(opt_s, string_maxlength) + 3); |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 205 | if (!str1) { |
| 206 | return fprintf(output, "???"); |
| 207 | } |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 208 | umovestr(proc, addr, MIN(opt_s, string_maxlength) + 1, str1); |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 209 | len = fprintf(output, "\""); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 210 | for (i = 0; len < MIN(opt_s, string_maxlength) + 1; i++) { |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 211 | if (str1[i]) { |
| 212 | len += display_char(str1[i]); |
| 213 | } else { |
| 214 | break; |
| 215 | } |
| 216 | } |
| 217 | len += fprintf(output, "\""); |
| 218 | if (str1[i] && (opt_s <= string_maxlength)) { |
| 219 | len += fprintf(output, "..."); |
| 220 | } |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 221 | for (i = 0; str1[i]; i++) { |
| 222 | if (str1[i] == '%') { |
| Juan Cespedes | d914a20 | 2004-11-10 00:15:33 +0100 | [diff] [blame] | 223 | int is_long = 0; |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 224 | while (1) { |
| Juan Cespedes | 5c3fe06 | 2004-06-14 18:08:37 +0200 | [diff] [blame] | 225 | unsigned char c = str1[++i]; |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 226 | if (c == '%') { |
| 227 | break; |
| 228 | } else if (!c) { |
| 229 | break; |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 230 | } else if (strchr("lzZtj", c)) { |
| Juan Cespedes | d914a20 | 2004-11-10 00:15:33 +0100 | [diff] [blame] | 231 | is_long++; |
| 232 | if (c == 'j') |
| 233 | is_long++; |
| Ian Wienand | 3219f32 | 2006-02-16 06:00:00 +0100 | [diff] [blame] | 234 | if (is_long > 1 |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 235 | && (sizeof(long) < sizeof(long long) |
| 236 | || proc->mask_32bit)) { |
| Juan Cespedes | d914a20 | 2004-11-10 00:15:33 +0100 | [diff] [blame] | 237 | len += fprintf(output, ", ..."); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 238 | str1[i + 1] = '\0'; |
| Juan Cespedes | d914a20 | 2004-11-10 00:15:33 +0100 | [diff] [blame] | 239 | break; |
| 240 | } |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 241 | } else if (c == 'd' || c == 'i') { |
| Ian Wienand | 9a2ad35 | 2006-02-20 22:44:45 +0100 | [diff] [blame] | 242 | if (!is_long || proc->mask_32bit) |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 243 | len += |
| 244 | fprintf(output, ", %d", |
| 245 | (int)gimme_arg(type, |
| 246 | proc, |
| 247 | ++arg_num)); |
| Juan Cespedes | d914a20 | 2004-11-10 00:15:33 +0100 | [diff] [blame] | 248 | else |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 249 | len += |
| 250 | fprintf(output, ", %ld", |
| 251 | gimme_arg(type, |
| 252 | proc, |
| 253 | ++arg_num)); |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 254 | break; |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 255 | } else if (c == 'u') { |
| Ian Wienand | 9a2ad35 | 2006-02-20 22:44:45 +0100 | [diff] [blame] | 256 | if (!is_long || proc->mask_32bit) |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 257 | len += |
| 258 | fprintf(output, ", %u", |
| 259 | (int)gimme_arg(type, |
| 260 | proc, |
| 261 | ++arg_num)); |
| Juan Cespedes | d914a20 | 2004-11-10 00:15:33 +0100 | [diff] [blame] | 262 | else |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 263 | len += |
| 264 | fprintf(output, ", %lu", |
| 265 | gimme_arg(type, |
| 266 | proc, |
| 267 | ++arg_num)); |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 268 | break; |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 269 | } else if (c == 'o') { |
| Ian Wienand | 9a2ad35 | 2006-02-20 22:44:45 +0100 | [diff] [blame] | 270 | if (!is_long || proc->mask_32bit) |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 271 | len += |
| 272 | fprintf(output, ", 0%o", |
| 273 | (int)gimme_arg(type, |
| 274 | proc, |
| 275 | ++arg_num)); |
| Juan Cespedes | d914a20 | 2004-11-10 00:15:33 +0100 | [diff] [blame] | 276 | else |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 277 | len += |
| 278 | fprintf(output, ", 0%lo", |
| 279 | gimme_arg(type, |
| 280 | proc, |
| 281 | ++arg_num)); |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 282 | break; |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 283 | } else if (c == 'x' || c == 'X') { |
| Ian Wienand | 9a2ad35 | 2006-02-20 22:44:45 +0100 | [diff] [blame] | 284 | if (!is_long || proc->mask_32bit) |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 285 | len += |
| 286 | fprintf(output, ", %#x", |
| 287 | (int)gimme_arg(type, |
| 288 | proc, |
| 289 | ++arg_num)); |
| Juan Cespedes | d914a20 | 2004-11-10 00:15:33 +0100 | [diff] [blame] | 290 | else |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 291 | len += |
| 292 | fprintf(output, ", %#lx", |
| 293 | gimme_arg(type, |
| 294 | proc, |
| 295 | ++arg_num)); |
| Juan Cespedes | d914a20 | 2004-11-10 00:15:33 +0100 | [diff] [blame] | 296 | break; |
| 297 | } else if (strchr("eEfFgGaACS", c) |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 298 | || (is_long |
| 299 | && (c == 'c' || c == 's'))) { |
| Juan Cespedes | d914a20 | 2004-11-10 00:15:33 +0100 | [diff] [blame] | 300 | len += fprintf(output, ", ..."); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 301 | str1[i + 1] = '\0'; |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 302 | break; |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 303 | } else if (c == 'c') { |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 304 | len += fprintf(output, ", '"); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 305 | len += |
| 306 | display_char((int) |
| 307 | gimme_arg(type, proc, |
| 308 | ++arg_num)); |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 309 | len += fprintf(output, "'"); |
| 310 | break; |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 311 | } else if (c == 's') { |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 312 | len += fprintf(output, ", "); |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 313 | len += |
| 314 | display_string(type, proc, |
| Steve Fink | 7bafff0 | 2006-08-07 04:50:42 +0200 | [diff] [blame] | 315 | (void *)gimme_arg(type, |
| 316 | proc, |
| 317 | ++arg_num), |
| Steve Fink | 6a48a6d | 2006-08-07 04:29:06 +0200 | [diff] [blame] | 318 | string_maxlength); |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 319 | break; |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 320 | } else if (c == 'p' || c == 'n') { |
| 321 | len += |
| 322 | fprintf(output, ", %p", |
| 323 | (void *)gimme_arg(type, |
| 324 | proc, |
| 325 | ++arg_num)); |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 326 | break; |
| Ian Wienand | 2d45b1a | 2006-02-20 22:48:07 +0100 | [diff] [blame] | 327 | } else if (c == '*') { |
| 328 | len += |
| 329 | fprintf(output, ", %d", |
| 330 | (int)gimme_arg(type, proc, |
| 331 | ++arg_num)); |
| Juan Cespedes | ac3db29 | 1998-04-25 14:31:58 +0200 | [diff] [blame] | 332 | } |
| 333 | } |
| 334 | } |
| 335 | } |
| 336 | free(str1); |
| 337 | return len; |
| 338 | } |