1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 /* 22 * Copyright 2007 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 */ 25 26 27 /* 28 * The Sun Studio and GCC (patched for opensolaris/illumos) compilers 29 * implement a argument saving scheme on amd64 via the -Wu,save-args or 30 * options. When the option is specified, INTEGER type function arguments 31 * passed via registers will be saved on the stack immediately after %rbp, and 32 * will not be modified through out the life of the routine. 33 * 34 * +--------+ 35 * %rbp --> | %rbp | 36 * +--------+ 37 * -0x8(%rbp) | %rdi | 38 * +--------+ 39 * -0x10(%rbp) | %rsi | 40 * +--------+ 41 * -0x18(%rbp) | %rdx | 42 * +--------+ 43 * -0x20(%rbp) | %rcx | 44 * +--------+ 45 * -0x28(%rbp) | %r8 | 46 * +--------+ 47 * -0x30(%rbp) | %r9 | 48 * +--------+ 49 * 50 * 51 * For example, for the following function, 52 * 53 * void 54 * foo(int a1, int a2, int a3, int a4, int a5, int a6, int a7) 55 * { 56 * ... 57 * } 58 * 59 * Disassembled code will look something like the following: 60 * 61 * pushq %rbp 62 * movq %rsp, %rbp 63 * subq $imm8, %rsp ** 64 * movq %rdi, -0x8(%rbp) 65 * movq %rsi, -0x10(%rbp) 66 * movq %rdx, -0x18(%rbp) 67 * movq %rcx, -0x20(%rbp) 68 * movq %r8, -0x28(%rbp) 69 * movq %r9, -0x30(%rbp) 70 * ... 71 * or 72 * pushq %rbp 73 * movq %rsp, %rbp 74 * subq $imm8, %rsp ** 75 * movq %r9, -0x30(%rbp) 76 * movq %r8, -0x28(%rbp) 77 * movq %rcx, -0x20(%rbp) 78 * movq %rdx, -0x18(%rbp) 79 * movq %rsi, -0x10(%rbp) 80 * movq %rdi, -0x8(%rbp) 81 * ... 82 * or 83 * pushq %rbp 84 * movq %rsp, %rbp 85 * pushq %rdi 86 * pushq %rsi 87 * pushq %rdx 88 * pushq %rcx 89 * pushq %r8 90 * pushq %r9 91 * 92 * **: The space being reserved is in addition to what the current 93 * function prolog already reserves. 94 * 95 * We loop through the first SAVEARGS_INSN_SEQ_LEN bytes of the function 96 * looking for each argument saving instruction we would expect to see. 97 * 98 * If there are odd number of arguments to a function, additional space is 99 * reserved on the stack to maintain 16-byte alignment. For example, 100 * 101 * argc == 0: no argument saving. 102 * argc == 3: save 3, but space for 4 is reserved 103 * argc == 7: save 6. 104 */ 105 106 #include <sys/sysmacros.h> 107 #include <sys/types.h> 108 #include <libdisasm.h> 109 #include <string.h> 110 111 #include <saveargs.h> 112 113 /* 114 * Size of the instruction sequence arrays. It should correspond to 115 * the maximum number of arguments passed via registers. 116 */ 117 #define INSTR_ARRAY_SIZE 6 118 119 #define INSTR1(ins, off) (ins[(off)]) 120 #define INSTR2(ins, off) (ins[(off)] + (ins[(off) + 1] << 8)) 121 #define INSTR3(ins, off) \ 122 (ins[(off)] + (ins[(off) + 1] << 8) + (ins[(off + 2)] << 16)) 123 #define INSTR4(ins, off) \ 124 (ins[(off)] + (ins[(off) + 1] << 8) + (ins[(off + 2)] << 16) + \ 125 (ins[(off) + 3] << 24)) 126 127 /* 128 * Sun Studio 10 patch implementation saves %rdi first; 129 * GCC 3.4.3 Sun branch implementation saves them in reverse order. 130 */ 131 static const uint32_t save_instr[INSTR_ARRAY_SIZE] = { 132 0xf87d8948, /* movq %rdi, -0x8(%rbp) */ 133 0xf0758948, /* movq %rsi, -0x10(%rbp) */ 134 0xe8558948, /* movq %rdx, -0x18(%rbp) */ 135 0xe04d8948, /* movq %rcx, -0x20(%rbp) */ 136 0xd845894c, /* movq %r8, -0x28(%rbp) */ 137 0xd04d894c /* movq %r9, -0x30(%rbp) */ 138 }; 139 140 static const uint16_t save_instr_push[] = { 141 0x57, /* pushq %rdi */ 142 0x56, /* pushq %rsi */ 143 0x52, /* pushq %rdx */ 144 0x51, /* pushq %rcx */ 145 0x5041, /* pushq %r8 */ 146 0x5141 /* pushq %r9 */ 147 }; 148 149 /* 150 * If the return type of a function is a structure greater than 16 bytes in 151 * size, %rdi will contain the address to which it should be stored, and 152 * arguments will begin at %rsi. Studio will push all of the normal argument 153 * registers anyway, GCC will start pushing at %rsi, so we need a separate 154 * pattern. 155 */ 156 static const uint32_t save_instr_sr[INSTR_ARRAY_SIZE-1] = { 157 0xf8758948, /* movq %rsi,-0x8(%rbp) */ 158 0xf0558948, /* movq %rdx,-0x10(%rbp) */ 159 0xe84d8948, /* movq %rcx,-0x18(%rbp) */ 160 0xe045894c, /* movq %r8,-0x20(%rbp) */ 161 0xd84d894c /* movq %r9,-0x28(%rbp) */ 162 }; 163 164 static const uint8_t save_fp_pushes[] = { 165 0x55, /* pushq %rbp */ 166 0xcc /* int $0x3 */ 167 }; 168 #define NUM_FP_PUSHES (sizeof (save_fp_pushes) / sizeof (save_fp_pushes[0])) 169 170 static const uint32_t save_fp_movs[] = { 171 0x00e58948, /* movq %rsp,%rbp, encoding 1 */ 172 0x00ec8b48, /* movq %rsp,%rbp, encoding 2 */ 173 }; 174 #define NUM_FP_MOVS (sizeof (save_fp_movs) / sizeof (save_fp_movs[0])) 175 176 typedef struct { 177 uint8_t *data; 178 size_t size; 179 } text_t; 180 181 static int 182 do_read(void *data, uint64_t addr, void *buf, size_t len) 183 { 184 text_t *t = data; 185 186 if (addr >= t->size) 187 return (-1); 188 189 len = MIN(len, t->size - addr); 190 191 (void) memcpy(buf, (char *)t->data + addr, len); 192 193 return (len); 194 } 195 196 /* ARGSUSED */ 197 int 198 do_lookup(void *data, uint64_t addr, char *buf, size_t buflen, uint64_t *start, 199 size_t *symlen) 200 { 201 /* We don't actually need lookup info */ 202 return (-1); 203 } 204 205 static int 206 instr_size(dis_handle_t *dhp, uint8_t *ins, unsigned int i, size_t size) 207 { 208 text_t t; 209 210 t.data = ins; 211 t.size = size; 212 213 dis_set_data(dhp, &t); 214 return (dis_instrlen(dhp, i)); 215 } 216 217 static boolean_t 218 has_saved_fp(dis_handle_t *dhp, uint8_t *ins, int size) 219 { 220 int i, j; 221 uint32_t n; 222 boolean_t found_push = B_FALSE; 223 ssize_t sz = 0; 224 225 for (i = 0; i < size; i += sz) { 226 if ((sz = instr_size(dhp, ins, i, size)) < 1) 227 return (B_FALSE); 228 229 if (found_push == B_FALSE) { 230 if (sz != 1) 231 continue; 232 233 n = INSTR1(ins, i); 234 for (j = 0; j <= NUM_FP_PUSHES; j++) 235 if (save_fp_pushes[j] == n) { 236 found_push = B_TRUE; 237 break; 238 } 239 } else { 240 if (sz != 3) 241 continue; 242 n = INSTR3(ins, i); 243 for (j = 0; j <= NUM_FP_MOVS; j++) 244 if (save_fp_movs[j] == n) 245 return (B_TRUE); 246 } 247 } 248 249 return (B_FALSE); 250 } 251 252 int 253 saveargs_has_args(uint8_t *ins, size_t size, uint_t argc, int start_index) 254 { 255 int i, j; 256 uint32_t n; 257 uint8_t found = 0; 258 ssize_t sz = 0; 259 dis_handle_t *dhp = NULL; 260 int ret = SAVEARGS_NO_ARGS; 261 262 argc = MIN((start_index + argc), INSTR_ARRAY_SIZE); 263 264 if ((dhp = dis_handle_create(DIS_X86_SIZE64, NULL, do_lookup, 265 do_read)) == NULL) 266 return (SAVEARGS_NO_ARGS); 267 268 if (!has_saved_fp(dhp, ins, size)) { 269 dis_handle_destroy(dhp); 270 return (SAVEARGS_NO_ARGS); 271 } 272 273 /* 274 * For each possible style of argument saving, walk the insn stream as 275 * we've been given it, and set bit N in 'found' if we find the 276 * instruction saving the Nth argument. 277 */ 278 279 /* 280 * Compare against regular implementation 281 */ 282 found = 0; 283 for (i = 0; i < size; i += sz) { 284 sz = instr_size(dhp, ins, i, size); 285 286 if (sz < 1) 287 break; 288 else if (sz != 4) 289 continue; 290 291 n = INSTR4(ins, i); 292 293 for (j = 0; j < argc; j++) { 294 if (n == save_instr[j]) { 295 found |= (1 << j); 296 297 if (found == ((1 << argc) - 1)) { 298 ret = start_index ? 299 SAVEARGS_STRUCT_ARGS : 300 SAVEARGS_TRAD_ARGS; 301 goto done; 302 } 303 304 break; 305 } 306 } 307 } 308 309 /* 310 * Compare against GCC push-based implementation 311 */ 312 found = 0; 313 for (i = 0; i < size; i += sz) { 314 if ((sz = instr_size(dhp, ins, i, size)) < 1) 315 break; 316 317 for (j = start_index; j < argc; j++) { 318 if (sz == 2) /* Two byte */ 319 n = INSTR2(ins, i); 320 else if (sz == 1) 321 n = INSTR1(ins, i); 322 else 323 continue; 324 325 if (n == save_instr_push[j]) { 326 found |= (1 << (j - start_index)); 327 328 if (found == 329 ((1 << (argc - start_index)) - 1)) { 330 ret = SAVEARGS_TRAD_ARGS; 331 goto done; 332 } 333 334 break; 335 } 336 } 337 } 338 339 /* 340 * Look for a GCC-style returned structure. 341 */ 342 found = 0; 343 if (start_index != 0) { 344 for (i = 0; i < size; i += sz) { 345 sz = instr_size(dhp, ins, i, size); 346 347 if (sz < 1) 348 break; 349 else if (sz != 4) 350 continue; 351 352 n = INSTR4(ins, i); 353 354 /* argc is inclusive of start_index, allow for that */ 355 for (j = 0; j < (argc - start_index); j++) { 356 if (n == save_instr_sr[j]) { 357 found |= (1 << j); 358 359 if (found == 360 ((1 << (argc - start_index)) - 1)) { 361 ret = SAVEARGS_TRAD_ARGS; 362 goto done; 363 } 364 365 break; 366 } 367 } 368 } 369 } 370 371 done: 372 dis_handle_destroy(dhp); 373 return (ret); 374 }