You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
641 lines
21 KiB
641 lines
21 KiB
commit 80f75320167acb66486124c6b03e715596e9c789 |
|
Author: Andreas Arnez <arnez@linux.vnet.ibm.com> |
|
Date: Mon Apr 27 11:38:46 2015 +0200 |
|
|
|
S390: Restructure s390_push_dummy_call |
|
|
|
Simplify the structure of s390_push_dummy_call and its various helper |
|
functions. This reduces the code and makes it easier to extend. The |
|
new code should be functionally equivalent to the old one, except that |
|
copies created by the caller are now always aligned on an 8-byte |
|
boundary. |
|
|
|
gdb/ChangeLog: |
|
|
|
* s390-linux-tdep.c |
|
(is_float_singleton): Remove function. Move the "singleton" part |
|
of the logic... |
|
(s390_effective_inner_type): ...here. New function. |
|
(is_float_like): Remove function. Inline its logic... |
|
(s390_function_arg_float): ...here. |
|
(is_pointer_like, is_integer_like, is_struct_like): Remove |
|
functions. Inline their logic... |
|
(s390_function_arg_integer): ...here. |
|
(s390_function_arg_pass_by_reference): Remove function. |
|
(extend_simple_arg): Remove function. |
|
(alignment_of): Remove function. |
|
(struct s390_arg_state): New structure. |
|
(s390_handle_arg): New function. |
|
(s390_push_dummy_call): Move parameter placement logic to the new |
|
function s390_handle_arg. Call it for calculating the stack area |
|
sizes first, and again for actually writing the parameters. |
|
|
|
### a/gdb/ChangeLog |
|
### b/gdb/ChangeLog |
|
## -1,5 +1,25 @@ |
|
2015-04-27 Andreas Arnez <arnez@linux.vnet.ibm.com> |
|
|
|
+ * s390-linux-tdep.c |
|
+ (is_float_singleton): Remove function. Move the "singleton" part |
|
+ of the logic... |
|
+ (s390_effective_inner_type): ...here. New function. |
|
+ (is_float_like): Remove function. Inline its logic... |
|
+ (s390_function_arg_float): ...here. |
|
+ (is_pointer_like, is_integer_like, is_struct_like): Remove |
|
+ functions. Inline their logic... |
|
+ (s390_function_arg_integer): ...here. |
|
+ (s390_function_arg_pass_by_reference): Remove function. |
|
+ (extend_simple_arg): Remove function. |
|
+ (alignment_of): Remove function. |
|
+ (struct s390_arg_state): New structure. |
|
+ (s390_handle_arg): New function. |
|
+ (s390_push_dummy_call): Move parameter placement logic to the new |
|
+ function s390_handle_arg. Call it for calculating the stack area |
|
+ sizes first, and again for actually writing the parameters. |
|
+ |
|
+2015-04-27 Andreas Arnez <arnez@linux.vnet.ibm.com> |
|
+ |
|
* s390-linux-tdep.c (is_power_of_two): Add comment. Return |
|
false if the argument is zero. |
|
|
|
Index: gdb-7.6.1/gdb/s390-tdep.c |
|
=================================================================== |
|
--- gdb-7.6.1.orig/gdb/s390-tdep.c 2016-02-21 22:27:47.198043364 +0100 |
|
+++ gdb-7.6.1/gdb/s390-tdep.c 2016-02-21 22:30:02.980113202 +0100 |
|
@@ -2546,99 +2546,40 @@ |
|
|
|
/* Dummy function calls. */ |
|
|
|
-/* Return non-zero if TYPE is an integer-like type, zero otherwise. |
|
- "Integer-like" types are those that should be passed the way |
|
- integers are: integers, enums, ranges, characters, and booleans. */ |
|
-static int |
|
-is_integer_like (struct type *type) |
|
-{ |
|
- enum type_code code = TYPE_CODE (type); |
|
+/* Unwrap any single-field structs in TYPE and return the effective |
|
+ "inner" type. E.g., yield "float" for all these cases: |
|
|
|
- return (code == TYPE_CODE_INT |
|
- || code == TYPE_CODE_ENUM |
|
- || code == TYPE_CODE_RANGE |
|
- || code == TYPE_CODE_CHAR |
|
- || code == TYPE_CODE_BOOL); |
|
-} |
|
+ float x; |
|
+ struct { float x }; |
|
+ struct { struct { float x; } x; }; |
|
+ struct { struct { struct { float x; } x; } x; }; */ |
|
|
|
-/* Return non-zero if TYPE is a pointer-like type, zero otherwise. |
|
- "Pointer-like" types are those that should be passed the way |
|
- pointers are: pointers and references. */ |
|
-static int |
|
-is_pointer_like (struct type *type) |
|
+static struct type * |
|
+s390_effective_inner_type (struct type *type) |
|
{ |
|
- enum type_code code = TYPE_CODE (type); |
|
- |
|
- return (code == TYPE_CODE_PTR |
|
- || code == TYPE_CODE_REF); |
|
+ while (TYPE_CODE (type) == TYPE_CODE_STRUCT |
|
+ && TYPE_NFIELDS (type) == 1) |
|
+ type = check_typedef (TYPE_FIELD_TYPE (type, 0)); |
|
+ return type; |
|
} |
|
|
|
+/* Return non-zero if TYPE should be passed like "float" or |
|
+ "double". */ |
|
|
|
-/* Return non-zero if TYPE is a `float singleton' or `double |
|
- singleton', zero otherwise. |
|
- |
|
- A `T singleton' is a struct type with one member, whose type is |
|
- either T or a `T singleton'. So, the following are all float |
|
- singletons: |
|
- |
|
- struct { float x }; |
|
- struct { struct { float x; } x; }; |
|
- struct { struct { struct { float x; } x; } x; }; |
|
- |
|
- ... and so on. |
|
- |
|
- All such structures are passed as if they were floats or doubles, |
|
- as the (revised) ABI says. */ |
|
static int |
|
-is_float_singleton (struct type *type) |
|
-{ |
|
- if (TYPE_CODE (type) == TYPE_CODE_STRUCT && TYPE_NFIELDS (type) == 1) |
|
- { |
|
- struct type *singleton_type = TYPE_FIELD_TYPE (type, 0); |
|
- CHECK_TYPEDEF (singleton_type); |
|
- |
|
- return (TYPE_CODE (singleton_type) == TYPE_CODE_FLT |
|
- || TYPE_CODE (singleton_type) == TYPE_CODE_DECFLOAT |
|
- || is_float_singleton (singleton_type)); |
|
- } |
|
- |
|
- return 0; |
|
-} |
|
- |
|
- |
|
-/* Return non-zero if TYPE is a struct-like type, zero otherwise. |
|
- "Struct-like" types are those that should be passed as structs are: |
|
- structs and unions. |
|
- |
|
- As an odd quirk, not mentioned in the ABI, GCC passes float and |
|
- double singletons as if they were a plain float, double, etc. (The |
|
- corresponding union types are handled normally.) So we exclude |
|
- those types here. *shrug* */ |
|
-static int |
|
-is_struct_like (struct type *type) |
|
+s390_function_arg_float (struct type *type) |
|
{ |
|
- enum type_code code = TYPE_CODE (type); |
|
- |
|
- return (code == TYPE_CODE_UNION |
|
- || (code == TYPE_CODE_STRUCT && ! is_float_singleton (type))); |
|
-} |
|
+ /* Note that long double as well as complex types are intentionally |
|
+ excluded. */ |
|
+ if (TYPE_LENGTH (type) > 8) |
|
+ return 0; |
|
|
|
+ /* A struct containing just a float or double is passed like a float |
|
+ or double. */ |
|
+ type = s390_effective_inner_type (type); |
|
|
|
-/* Return non-zero if TYPE is a float-like type, zero otherwise. |
|
- "Float-like" types are those that should be passed as |
|
- floating-point values are. |
|
- |
|
- You'd think this would just be floats, doubles, long doubles, etc. |
|
- But as an odd quirk, not mentioned in the ABI, GCC passes float and |
|
- double singletons as if they were a plain float, double, etc. (The |
|
- corresponding union types are handled normally.) So we include |
|
- those types here. *shrug* */ |
|
-static int |
|
-is_float_like (struct type *type) |
|
-{ |
|
return (TYPE_CODE (type) == TYPE_CODE_FLT |
|
- || TYPE_CODE (type) == TYPE_CODE_DECFLOAT |
|
- || is_float_singleton (type)); |
|
+ || TYPE_CODE (type) == TYPE_CODE_DECFLOAT); |
|
} |
|
|
|
/* Determine whether N is a power of two. */ |
|
@@ -2649,101 +2590,172 @@ |
|
return n && ((n & (n - 1)) == 0); |
|
} |
|
|
|
-/* Return non-zero if TYPE should be passed as a pointer to a copy, |
|
- zero otherwise. */ |
|
-static int |
|
-s390_function_arg_pass_by_reference (struct type *type) |
|
-{ |
|
- if (TYPE_LENGTH (type) > 8) |
|
- return 1; |
|
- |
|
- return (is_struct_like (type) && !is_power_of_two (TYPE_LENGTH (type))) |
|
- || TYPE_CODE (type) == TYPE_CODE_COMPLEX |
|
- || (TYPE_CODE (type) == TYPE_CODE_ARRAY && TYPE_VECTOR (type)); |
|
-} |
|
- |
|
-/* Return non-zero if TYPE should be passed in a float register |
|
- if possible. */ |
|
-static int |
|
-s390_function_arg_float (struct type *type) |
|
-{ |
|
- if (TYPE_LENGTH (type) > 8) |
|
- return 0; |
|
- |
|
- return is_float_like (type); |
|
-} |
|
+/* For an argument whose type is TYPE and which is not passed like a |
|
+ float, return non-zero if it should be passed like "int" or "long |
|
+ long". */ |
|
|
|
-/* Return non-zero if TYPE should be passed in an integer register |
|
- (or a pair of integer registers) if possible. */ |
|
static int |
|
s390_function_arg_integer (struct type *type) |
|
{ |
|
+ enum type_code code = TYPE_CODE (type); |
|
+ |
|
if (TYPE_LENGTH (type) > 8) |
|
return 0; |
|
|
|
- return is_integer_like (type) |
|
- || is_pointer_like (type) |
|
- || (is_struct_like (type) && is_power_of_two (TYPE_LENGTH (type))); |
|
-} |
|
- |
|
-/* Return ARG, a `SIMPLE_ARG', sign-extended or zero-extended to a full |
|
- word as required for the ABI. */ |
|
-static LONGEST |
|
-extend_simple_arg (struct gdbarch *gdbarch, struct value *arg) |
|
-{ |
|
- enum bfd_endian byte_order = gdbarch_byte_order (gdbarch); |
|
- struct type *type = check_typedef (value_type (arg)); |
|
+ if (code == TYPE_CODE_INT |
|
+ || code == TYPE_CODE_ENUM |
|
+ || code == TYPE_CODE_RANGE |
|
+ || code == TYPE_CODE_CHAR |
|
+ || code == TYPE_CODE_BOOL |
|
+ || code == TYPE_CODE_PTR |
|
+ || code == TYPE_CODE_REF) |
|
+ return 1; |
|
|
|
- /* Even structs get passed in the least significant bits of the |
|
- register / memory word. It's not really right to extract them as |
|
- an integer, but it does take care of the extension. */ |
|
- if (TYPE_UNSIGNED (type)) |
|
- return extract_unsigned_integer (value_contents (arg), |
|
- TYPE_LENGTH (type), byte_order); |
|
- else |
|
- return extract_signed_integer (value_contents (arg), |
|
- TYPE_LENGTH (type), byte_order); |
|
+ return ((code == TYPE_CODE_UNION || code == TYPE_CODE_STRUCT) |
|
+ && is_power_of_two (TYPE_LENGTH (type))); |
|
} |
|
|
|
+/* Argument passing state: Internal data structure passed to helper |
|
+ routines of s390_push_dummy_call. */ |
|
|
|
-/* Return the alignment required by TYPE. */ |
|
-static int |
|
-alignment_of (struct type *type) |
|
+struct s390_arg_state |
|
+ { |
|
+ /* Register cache, or NULL, if we are in "preparation mode". */ |
|
+ struct regcache *regcache; |
|
+ /* Next available general/floating-point register for argument |
|
+ passing. */ |
|
+ int gr, fr; |
|
+ /* Current pointer to copy area (grows downwards). */ |
|
+ CORE_ADDR copy; |
|
+ /* Current pointer to parameter area (grows upwards). */ |
|
+ CORE_ADDR argp; |
|
+ }; |
|
+ |
|
+/* Prepare one argument ARG for a dummy call and update the argument |
|
+ passing state AS accordingly. If the regcache field in AS is set, |
|
+ operate in "write mode" and write ARG into the inferior. Otherwise |
|
+ run "preparation mode" and skip all updates to the inferior. */ |
|
+ |
|
+static void |
|
+s390_handle_arg (struct s390_arg_state *as, struct value *arg, |
|
+ struct gdbarch_tdep *tdep, int word_size, |
|
+ enum bfd_endian byte_order) |
|
{ |
|
- int alignment; |
|
+ struct type *type = check_typedef (value_type (arg)); |
|
+ ULONGEST length = TYPE_LENGTH (type); |
|
+ int write_mode = as->regcache != NULL; |
|
|
|
- if (is_integer_like (type) |
|
- || is_pointer_like (type) |
|
- || TYPE_CODE (type) == TYPE_CODE_FLT |
|
- || TYPE_CODE (type) == TYPE_CODE_DECFLOAT) |
|
- alignment = TYPE_LENGTH (type); |
|
- else if (TYPE_CODE (type) == TYPE_CODE_STRUCT |
|
- || TYPE_CODE (type) == TYPE_CODE_UNION) |
|
+ if (s390_function_arg_float (type)) |
|
+ { |
|
+ /* The GNU/Linux for S/390 ABI uses FPRs 0 and 2 to pass |
|
+ arguments. The GNU/Linux for zSeries ABI uses 0, 2, 4, and |
|
+ 6. */ |
|
+ if (as->fr <= (tdep->abi == ABI_LINUX_S390 ? 2 : 6)) |
|
+ { |
|
+ /* When we store a single-precision value in an FP register, |
|
+ it occupies the leftmost bits. */ |
|
+ if (write_mode) |
|
+ regcache_cooked_write_part (as->regcache, |
|
+ S390_F0_REGNUM + as->fr, |
|
+ 0, length, |
|
+ value_contents (arg)); |
|
+ as->fr += 2; |
|
+ } |
|
+ else |
|
+ { |
|
+ /* When we store a single-precision value in a stack slot, |
|
+ it occupies the rightmost bits. */ |
|
+ as->argp = align_up (as->argp + length, word_size); |
|
+ if (write_mode) |
|
+ write_memory (as->argp - length, value_contents (arg), |
|
+ length); |
|
+ } |
|
+ } |
|
+ else if (s390_function_arg_integer (type) && length <= word_size) |
|
{ |
|
- int i; |
|
+ ULONGEST val; |
|
|
|
- alignment = 1; |
|
- for (i = 0; i < TYPE_NFIELDS (type); i++) |
|
+ if (write_mode) |
|
{ |
|
- int field_alignment |
|
- = alignment_of (check_typedef (TYPE_FIELD_TYPE (type, i))); |
|
- |
|
- if (field_alignment > alignment) |
|
- alignment = field_alignment; |
|
+ /* Place value in least significant bits of the register or |
|
+ memory word and sign- or zero-extend to full word size. |
|
+ This also applies to a struct or union. */ |
|
+ val = TYPE_UNSIGNED (type) |
|
+ ? extract_unsigned_integer (value_contents (arg), |
|
+ length, byte_order) |
|
+ : extract_signed_integer (value_contents (arg), |
|
+ length, byte_order); |
|
+ } |
|
+ |
|
+ if (as->gr <= 6) |
|
+ { |
|
+ if (write_mode) |
|
+ regcache_cooked_write_unsigned (as->regcache, |
|
+ S390_R0_REGNUM + as->gr, |
|
+ val); |
|
+ as->gr++; |
|
+ } |
|
+ else |
|
+ { |
|
+ if (write_mode) |
|
+ write_memory_unsigned_integer (as->argp, word_size, |
|
+ byte_order, val); |
|
+ as->argp += word_size; |
|
} |
|
} |
|
+ else if (s390_function_arg_integer (type) && length == 8) |
|
+ { |
|
+ if (as->gr <= 5) |
|
+ { |
|
+ if (write_mode) |
|
+ { |
|
+ regcache_cooked_write (as->regcache, |
|
+ S390_R0_REGNUM + as->gr, |
|
+ value_contents (arg)); |
|
+ regcache_cooked_write (as->regcache, |
|
+ S390_R0_REGNUM + as->gr + 1, |
|
+ value_contents (arg) + word_size); |
|
+ } |
|
+ as->gr += 2; |
|
+ } |
|
+ else |
|
+ { |
|
+ /* If we skipped r6 because we couldn't fit a DOUBLE_ARG |
|
+ in it, then don't go back and use it again later. */ |
|
+ as->gr = 7; |
|
+ |
|
+ if (write_mode) |
|
+ write_memory (as->argp, value_contents (arg), length); |
|
+ as->argp += length; |
|
+ } |
|
+ } |
|
else |
|
- alignment = 1; |
|
- |
|
- /* Check that everything we ever return is a power of two. Lots of |
|
- code doesn't want to deal with aligning things to arbitrary |
|
- boundaries. */ |
|
- gdb_assert ((alignment & (alignment - 1)) == 0); |
|
- |
|
- return alignment; |
|
+ { |
|
+ /* This argument type is never passed in registers. Place the |
|
+ value in the copy area and pass a pointer to it. Use 8-byte |
|
+ alignment as a conservative assumption. */ |
|
+ as->copy = align_down (as->copy - length, 8); |
|
+ if (write_mode) |
|
+ write_memory (as->copy, value_contents (arg), length); |
|
+ |
|
+ if (as->gr <= 6) |
|
+ { |
|
+ if (write_mode) |
|
+ regcache_cooked_write_unsigned (as->regcache, |
|
+ S390_R0_REGNUM + as->gr, |
|
+ as->copy); |
|
+ as->gr++; |
|
+ } |
|
+ else |
|
+ { |
|
+ if (write_mode) |
|
+ write_memory_unsigned_integer (as->argp, word_size, |
|
+ byte_order, as->copy); |
|
+ as->argp += word_size; |
|
+ } |
|
+ } |
|
} |
|
|
|
- |
|
/* Put the actual parameter values pointed to by ARGS[0..NARGS-1] in |
|
place to be passed to a function, as specified by the "GNU/Linux |
|
for S/390 ELF Application Binary Interface Supplement". |
|
@@ -2758,6 +2770,7 @@ |
|
|
|
Our caller has taken care of any type promotions needed to satisfy |
|
prototypes or the old K&R argument-passing rules. */ |
|
+ |
|
static CORE_ADDR |
|
s390_push_dummy_call (struct gdbarch *gdbarch, struct value *function, |
|
struct regcache *regcache, CORE_ADDR bp_addr, |
|
@@ -2768,151 +2781,48 @@ |
|
int word_size = gdbarch_ptr_bit (gdbarch) / 8; |
|
enum bfd_endian byte_order = gdbarch_byte_order (gdbarch); |
|
int i; |
|
+ struct s390_arg_state arg_state, arg_prep; |
|
+ CORE_ADDR param_area_start, new_sp; |
|
|
|
- /* If the i'th argument is passed as a reference to a copy, then |
|
- copy_addr[i] is the address of the copy we made. */ |
|
- CORE_ADDR *copy_addr = alloca (nargs * sizeof (CORE_ADDR)); |
|
+ arg_prep.copy = sp; |
|
+ arg_prep.gr = struct_return ? 3 : 2; |
|
+ arg_prep.fr = 0; |
|
+ arg_prep.argp = 0; |
|
+ arg_prep.regcache = NULL; |
|
|
|
- /* Reserve space for the reference-to-copy area. */ |
|
- for (i = 0; i < nargs; i++) |
|
- { |
|
- struct value *arg = args[i]; |
|
- struct type *type = check_typedef (value_type (arg)); |
|
+ /* Initialize arg_state for "preparation mode". */ |
|
+ arg_state = arg_prep; |
|
|
|
- if (s390_function_arg_pass_by_reference (type)) |
|
- { |
|
- sp -= TYPE_LENGTH (type); |
|
- sp = align_down (sp, alignment_of (type)); |
|
- copy_addr[i] = sp; |
|
- } |
|
- } |
|
+ /* Update arg_state.copy with the start of the reference-to-copy area |
|
+ and arg_state.argp with the size of the parameter area. */ |
|
+ for (i = 0; i < nargs; i++) |
|
+ s390_handle_arg (&arg_state, args[i], tdep, word_size, byte_order); |
|
|
|
- /* Reserve space for the parameter area. As a conservative |
|
- simplification, we assume that everything will be passed on the |
|
- stack. Since every argument larger than 8 bytes will be |
|
- passed by reference, we use this simple upper bound. */ |
|
- sp -= nargs * 8; |
|
- |
|
- /* After all that, make sure it's still aligned on an eight-byte |
|
- boundary. */ |
|
- sp = align_down (sp, 8); |
|
+ param_area_start = align_down (arg_state.copy - arg_state.argp, 8); |
|
|
|
/* Allocate the standard frame areas: the register save area, the |
|
- word reserved for the compiler (which seems kind of meaningless), |
|
- and the back chain pointer. */ |
|
- sp -= 16*word_size + 32; |
|
- |
|
- /* Now we have the final SP value. Make sure we didn't underflow; |
|
- on 31-bit, this would result in addresses with the high bit set, |
|
- which causes confusion elsewhere. Note that if we error out |
|
- here, stack and registers remain untouched. */ |
|
- if (gdbarch_addr_bits_remove (gdbarch, sp) != sp) |
|
+ word reserved for the compiler, and the back chain pointer. */ |
|
+ new_sp = param_area_start - (16 * word_size + 32); |
|
+ |
|
+ /* Now we have the final stack pointer. Make sure we didn't |
|
+ underflow; on 31-bit, this would result in addresses with the |
|
+ high bit set, which causes confusion elsewhere. Note that if we |
|
+ error out here, stack and registers remain untouched. */ |
|
+ if (gdbarch_addr_bits_remove (gdbarch, new_sp) != new_sp) |
|
error (_("Stack overflow")); |
|
|
|
+ /* Pass the structure return address in general register 2. */ |
|
+ if (struct_return) |
|
+ regcache_cooked_write_unsigned (regcache, S390_R2_REGNUM, struct_addr); |
|
+ |
|
+ /* Initialize arg_state for "write mode". */ |
|
+ arg_state = arg_prep; |
|
+ arg_state.argp = param_area_start; |
|
+ arg_state.regcache = regcache; |
|
|
|
- /* Finally, place the actual parameters, working from SP towards |
|
- higher addresses. The code above is supposed to reserve enough |
|
- space for this. */ |
|
- { |
|
- int fr = 0; |
|
- int gr = 2; |
|
- CORE_ADDR starg = sp + 16*word_size + 32; |
|
- |
|
- /* A struct is returned using general register 2. */ |
|
- if (struct_return) |
|
- { |
|
- regcache_cooked_write_unsigned (regcache, S390_R0_REGNUM + gr, |
|
- struct_addr); |
|
- gr++; |
|
- } |
|
- |
|
- for (i = 0; i < nargs; i++) |
|
- { |
|
- struct value *arg = args[i]; |
|
- struct type *type = check_typedef (value_type (arg)); |
|
- ULONGEST length = TYPE_LENGTH (type); |
|
- |
|
- if (s390_function_arg_pass_by_reference (type)) |
|
- { |
|
- /* Actually copy the argument contents to the stack slot |
|
- that was reserved above. */ |
|
- write_memory (copy_addr[i], value_contents (arg), length); |
|
- |
|
- if (gr <= 6) |
|
- { |
|
- regcache_cooked_write_unsigned (regcache, S390_R0_REGNUM + gr, |
|
- copy_addr[i]); |
|
- gr++; |
|
- } |
|
- else |
|
- { |
|
- write_memory_unsigned_integer (starg, word_size, byte_order, |
|
- copy_addr[i]); |
|
- starg += word_size; |
|
- } |
|
- } |
|
- else if (s390_function_arg_float (type)) |
|
- { |
|
- /* The GNU/Linux for S/390 ABI uses FPRs 0 and 2 to pass arguments, |
|
- the GNU/Linux for zSeries ABI uses 0, 2, 4, and 6. */ |
|
- if (fr <= (tdep->abi == ABI_LINUX_S390 ? 2 : 6)) |
|
- { |
|
- /* When we store a single-precision value in an FP register, |
|
- it occupies the leftmost bits. */ |
|
- regcache_cooked_write_part (regcache, S390_F0_REGNUM + fr, |
|
- 0, length, value_contents (arg)); |
|
- fr += 2; |
|
- } |
|
- else |
|
- { |
|
- /* When we store a single-precision value in a stack slot, |
|
- it occupies the rightmost bits. */ |
|
- starg = align_up (starg + length, word_size); |
|
- write_memory (starg - length, value_contents (arg), length); |
|
- } |
|
- } |
|
- else if (s390_function_arg_integer (type) && length <= word_size) |
|
- { |
|
- if (gr <= 6) |
|
- { |
|
- /* Integer arguments are always extended to word size. */ |
|
- regcache_cooked_write_signed (regcache, S390_R0_REGNUM + gr, |
|
- extend_simple_arg (gdbarch, |
|
- arg)); |
|
- gr++; |
|
- } |
|
- else |
|
- { |
|
- /* Integer arguments are always extended to word size. */ |
|
- write_memory_signed_integer (starg, word_size, byte_order, |
|
- extend_simple_arg (gdbarch, arg)); |
|
- starg += word_size; |
|
- } |
|
- } |
|
- else if (s390_function_arg_integer (type) && length == 2*word_size) |
|
- { |
|
- if (gr <= 5) |
|
- { |
|
- regcache_cooked_write (regcache, S390_R0_REGNUM + gr, |
|
- value_contents (arg)); |
|
- regcache_cooked_write (regcache, S390_R0_REGNUM + gr + 1, |
|
- value_contents (arg) + word_size); |
|
- gr += 2; |
|
- } |
|
- else |
|
- { |
|
- /* If we skipped r6 because we couldn't fit a DOUBLE_ARG |
|
- in it, then don't go back and use it again later. */ |
|
- gr = 7; |
|
- |
|
- write_memory (starg, value_contents (arg), length); |
|
- starg += length; |
|
- } |
|
- } |
|
- else |
|
- internal_error (__FILE__, __LINE__, _("unknown argument type")); |
|
- } |
|
- } |
|
+ /* Write all parameters. */ |
|
+ for (i = 0; i < nargs; i++) |
|
+ s390_handle_arg (&arg_state, args[i], tdep, word_size, byte_order); |
|
|
|
/* Store return PSWA. In 31-bit mode, keep addressing mode bit. */ |
|
if (word_size == 4) |
|
@@ -2924,11 +2834,11 @@ |
|
regcache_cooked_write_unsigned (regcache, S390_RETADDR_REGNUM, bp_addr); |
|
|
|
/* Store updated stack pointer. */ |
|
- regcache_cooked_write_unsigned (regcache, S390_SP_REGNUM, sp); |
|
+ regcache_cooked_write_unsigned (regcache, S390_SP_REGNUM, new_sp); |
|
|
|
/* We need to return the 'stack part' of the frame ID, |
|
which is actually the top of the register save area. */ |
|
- return sp + 16*word_size + 32; |
|
+ return param_area_start; |
|
} |
|
|
|
/* Assuming THIS_FRAME is a dummy, return the frame ID of that
|
|
|