1/* mips16 floating point support code 2 Copyright (C) 1996-2018 Free Software Foundation, Inc. 3 Contributed by Cygnus Support 4 5This file is free software; you can redistribute it and/or modify it 6under the terms of the GNU General Public License as published by the 7Free Software Foundation; either version 3, or (at your option) any 8later version. 9 10This file is distributed in the hope that it will be useful, but 11WITHOUT ANY WARRANTY; without even the implied warranty of 12MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 13General Public License for more details. 14 15Under Section 7 of GPL version 3, you are granted additional 16permissions described in the GCC Runtime Library Exception, version 173.1, as published by the Free Software Foundation. 18 19You should have received a copy of the GNU General Public License and 20a copy of the GCC Runtime Library Exception along with this program; 21see the files COPYING3 and COPYING.RUNTIME respectively. If not, see 22<http://www.gnu.org/licenses/>. */ 23 24#include "auto-host.h" 25 26#if defined(__mips_micromips) || defined(__mips_soft_float) \ 27 || __mips_isa_rev >= 6 28 /* Do nothing because this code is only needed when linking 29 against mips16 hard-float objects. Neither micromips code 30 nor soft-float nor MIPS R6 code can be linked against mips16 31 hard-float objects so we do not need these routines when 32 building libgcc for those cases. */ 33#else 34 35#if defined(HAVE_AS_MODULE) 36#if __mips_fpr == 32 37 .module fp=32 38#elif __mips_fpr == 0 39 .module fp=xx 40#elif __mips_fpr == 64 41 .module fp=64 42#endif 43#endif 44 45/* This file contains mips16 floating point support functions. These 46 functions are called by mips16 code to handle floating point when 47 -msoft-float is not used. They accept the arguments and return 48 values using the soft-float calling convention, but do the actual 49 operation using the hard floating point instructions. */ 50 51#if defined _MIPS_SIM && (_MIPS_SIM == _ABIO32 || _MIPS_SIM == _ABIO64) 52 53/* This file contains 32-bit assembly code. */ 54 .set nomips16 55 56/* Start a function. */ 57 58#define STARTFN(NAME) .globl NAME; .ent NAME; NAME: 59 60/* Finish a function. */ 61 62#define ENDFN(NAME) .end NAME 63 64/* ARG1 65 The FPR that holds the first floating-point argument. 66 67 ARG2 68 The FPR that holds the second floating-point argument. 69 70 RET 71 The FPR that holds a floating-point return value. */ 72 73#define RET $f0 74#define ARG1 $f12 75#ifdef __mips64 76#define ARG2 $f13 77#else 78#define ARG2 $f14 79#endif 80 81/* Set 64-bit register GPR so that its high 32 bits contain HIGH_FPR 82 and so that its low 32 bits contain LOW_FPR. */ 83#define MERGE_GPRf(GPR, HIGH_FPR, LOW_FPR) \ 84 .set noat; \ 85 mfc1 $1, LOW_FPR; \ 86 mfc1 GPR, HIGH_FPR; \ 87 dsll $1, $1, 32; \ 88 dsll GPR, GPR, 32; \ 89 dsrl $1, $1, 32; \ 90 or GPR, GPR, $1; \ 91 .set at 92 93/* Move the high 32 bits of GPR to HIGH_FPR and the low 32 bits of 94 GPR to LOW_FPR. */ 95#define MERGE_GPRt(GPR, HIGH_FPR, LOW_FPR) \ 96 .set noat; \ 97 dsrl $1, GPR, 32; \ 98 mtc1 GPR, LOW_FPR; \ 99 mtc1 $1, HIGH_FPR; \ 100 .set at 101 102/* Jump to T, and use "OPCODE, OP2" to implement a delayed move. */ 103#define DELAYt(T, OPCODE, OP2) \ 104 .set noreorder; \ 105 jr T; \ 106 OPCODE, OP2; \ 107 .set reorder 108 109#if __mips >= 4 110/* Coprocessor moves are interlocked from the MIPS IV ISA up. */ 111#define DELAYf(T, OPCODE, OP2) DELAYt (T, OPCODE, OP2) 112#else 113/* Use "OPCODE. OP2" and jump to T. */ 114#define DELAYf(T, OPCODE, OP2) OPCODE, OP2; jr T 115#endif 116 117/* MOVE_SF_BYTE0(D) 118 Move the first single-precision floating-point argument between 119 GPRs and FPRs. 120 121 MOVE_SI_BYTE0(D) 122 Likewise the first single-precision integer argument. 123 124 MOVE_SF_BYTE4(D) 125 Move the second single-precision floating-point argument between 126 GPRs and FPRs, given that the first argument occupies 4 bytes. 127 128 MOVE_SF_BYTE8(D) 129 Move the second single-precision floating-point argument between 130 GPRs and FPRs, given that the first argument occupies 8 bytes. 131 132 MOVE_DF_BYTE0(D) 133 Move the first double-precision floating-point argument between 134 GPRs and FPRs. 135 136 MOVE_DF_BYTE8(D) 137 Likewise the second double-precision floating-point argument. 138 139 MOVE_SF_RET(D, T) 140 Likewise a single-precision floating-point return value, 141 then jump to T. 142 143 MOVE_SC_RET(D, T) 144 Likewise a complex single-precision floating-point return value. 145 146 MOVE_DF_RET(D, T) 147 Likewise a double-precision floating-point return value. 148 149 MOVE_DC_RET(D, T) 150 Likewise a complex double-precision floating-point return value. 151 152 MOVE_SI_RET(D, T) 153 Likewise a single-precision integer return value. 154 155 The D argument is "t" to move to FPRs and "f" to move from FPRs. 156 The return macros may assume that the target of the jump does not 157 use a floating-point register. */ 158 159#define MOVE_SF_RET(D, T) DELAY##D (T, m##D##c1 $2,$f0) 160#define MOVE_SI_RET(D, T) DELAY##D (T, m##D##c1 $2,$f0) 161 162#if defined(__mips64) && defined(__MIPSEB__) 163#define MOVE_SC_RET(D, T) MERGE_GPR##D ($2, $f0, $f1); jr T 164#elif defined(__mips64) 165/* The high 32 bits of $2 correspond to the second word in memory; 166 i.e. the imaginary part. */ 167#define MOVE_SC_RET(D, T) MERGE_GPR##D ($2, $f1, $f0); jr T 168#else 169#define MOVE_SC_RET(D, T) m##D##c1 $2,$f0; DELAY##D (T, m##D##c1 $3,$f2) 170#endif 171 172#if defined(__mips64) 173#define MOVE_SF_BYTE0(D) m##D##c1 $4,$f12 174#define MOVE_SF_BYTE4(D) m##D##c1 $5,$f13 175#define MOVE_SF_BYTE8(D) m##D##c1 $5,$f13 176#else 177#define MOVE_SF_BYTE0(D) m##D##c1 $4,$f12 178#define MOVE_SF_BYTE4(D) m##D##c1 $5,$f14 179#define MOVE_SF_BYTE8(D) m##D##c1 $6,$f14 180#endif 181#define MOVE_SI_BYTE0(D) MOVE_SF_BYTE0(D) 182 183#if defined(__mips64) 184#define MOVE_DF_BYTE0(D) dm##D##c1 $4,$f12 185#define MOVE_DF_BYTE8(D) dm##D##c1 $5,$f13 186#define MOVE_DF_RET(D, T) DELAY##D (T, dm##D##c1 $2,$f0) 187#define MOVE_DC_RET(D, T) dm##D##c1 $3,$f1; MOVE_DF_RET (D, T) 188#elif __mips_fpr != 32 && __mips_isa_rev >= 2 && defined(__MIPSEB__) 189#define MOVE_DF_BYTE0(D) m##D##c1 $5,$f12; m##D##hc1 $4,$f12 190#define MOVE_DF_BYTE8(D) m##D##c1 $7,$f14; m##D##hc1 $6,$f14 191#define MOVE_DF_RET(D, T) m##D##c1 $3,$f0; DELAY##D (T, m##D##hc1 $2,$f0) 192#define MOVE_DC_RET(D, T) m##D##c1 $5,$f2; m##D##hc1 $4,$f2; MOVE_DF_RET (D, T) 193#elif __mips_fpr != 32 && __mips_isa_rev >= 2 194#define MOVE_DF_BYTE0(D) m##D##c1 $4,$f12; m##D##hc1 $5,$f12 195#define MOVE_DF_BYTE8(D) m##D##c1 $6,$f14; m##D##hc1 $7,$f14 196#define MOVE_DF_RET(D, T) m##D##c1 $2,$f0; DELAY##D (T, m##D##hc1 $3,$f0) 197#define MOVE_DC_RET(D, T) m##D##c1 $4,$f2; m##D##hc1 $5,$f2; MOVE_DF_RET (D, T) 198#elif __mips_fpr == 0 199#define MOVE_DF_BYTE0t sw $4, 0($29); sw $5, 4($29); ldc1 $f12, 0($29) 200#define MOVE_DF_BYTE0f sdc1 $f12, 0($29); lw $4, 0($29); lw $5, 4($29) 201#define MOVE_DF_BYTE0(D) MOVE_DF_BYTE0##D 202#define MOVE_DF_BYTE8t sw $6, 8($29); sw $7, 12($29); ldc1 $f14, 8($29) 203#define MOVE_DF_BYTE8f sdc1 $f14, 8($29); lw $6, 8($29); lw $7, 12($29) 204#define MOVE_DF_BYTE8(D) MOVE_DF_BYTE8##D 205#define MOVE_DF_RETt(T) sw $2, 0($29); sw $3, 4($29); DELAYt (T, ldc1 $f0, 0($29)) 206#define MOVE_DF_RETf(T) sdc1 $f0, 0($29); lw $2, 0($29); DELAYf (T, lw $3, 4($29)) 207#define MOVE_DF_RET(D, T) MOVE_DF_RET##D(T) 208#define MOVE_DC_RETt(T) sw $4, 8($29); sw $5, 12($29); ldc1 $f2, 8($29); MOVE_DF_RETt(T) 209#define MOVE_DC_RETf(T) sdc1 $f2, 8($29); lw $4, 8($29); lw $5, 12($29); MOVE_DF_RETf(T) 210#define MOVE_DC_RET(D, T) MOVE_DF_RET##D(T) 211#elif defined(__MIPSEB__) 212/* FPRs are little-endian. */ 213#define MOVE_DF_BYTE0(D) m##D##c1 $4,$f13; m##D##c1 $5,$f12 214#define MOVE_DF_BYTE8(D) m##D##c1 $6,$f15; m##D##c1 $7,$f14 215#define MOVE_DF_RET(D, T) m##D##c1 $2,$f1; DELAY##D (T, m##D##c1 $3,$f0) 216#define MOVE_DC_RET(D, T) m##D##c1 $4,$f3; m##D##c1 $5,$f2; MOVE_DF_RET (D, T) 217#else 218#define MOVE_DF_BYTE0(D) m##D##c1 $4,$f12; m##D##c1 $5,$f13 219#define MOVE_DF_BYTE8(D) m##D##c1 $6,$f14; m##D##c1 $7,$f15 220#define MOVE_DF_RET(D, T) m##D##c1 $2,$f0; DELAY##D (T, m##D##c1 $3,$f1) 221#define MOVE_DC_RET(D, T) m##D##c1 $4,$f2; m##D##c1 $5,$f3; MOVE_DF_RET (D, T) 222#endif 223 224/* Single-precision math. */ 225 226/* Define a function NAME that loads two single-precision values, 227 performs FPU operation OPCODE on them, and returns the single- 228 precision result. */ 229 230#define OPSF3(NAME, OPCODE) \ 231STARTFN (NAME); \ 232 MOVE_SF_BYTE0 (t); \ 233 MOVE_SF_BYTE4 (t); \ 234 OPCODE RET,ARG1,ARG2; \ 235 MOVE_SF_RET (f, $31); \ 236 ENDFN (NAME) 237 238#ifdef L_m16addsf3 239OPSF3 (__mips16_addsf3, add.s) 240#endif 241#ifdef L_m16subsf3 242OPSF3 (__mips16_subsf3, sub.s) 243#endif 244#ifdef L_m16mulsf3 245OPSF3 (__mips16_mulsf3, mul.s) 246#endif 247#ifdef L_m16divsf3 248OPSF3 (__mips16_divsf3, div.s) 249#endif 250 251/* Define a function NAME that loads a single-precision value, 252 performs FPU operation OPCODE on it, and returns the single- 253 precision result. */ 254 255#define OPSF2(NAME, OPCODE) \ 256STARTFN (NAME); \ 257 MOVE_SF_BYTE0 (t); \ 258 OPCODE RET,ARG1; \ 259 MOVE_SF_RET (f, $31); \ 260 ENDFN (NAME) 261 262#ifdef L_m16negsf2 263OPSF2 (__mips16_negsf2, neg.s) 264#endif 265#ifdef L_m16abssf2 266OPSF2 (__mips16_abssf2, abs.s) 267#endif 268 269/* Single-precision comparisons. */ 270 271/* Define a function NAME that loads two single-precision values, 272 performs floating point comparison OPCODE, and returns TRUE or 273 FALSE depending on the result. */ 274 275#define CMPSF(NAME, OPCODE, TRUE, FALSE) \ 276STARTFN (NAME); \ 277 MOVE_SF_BYTE0 (t); \ 278 MOVE_SF_BYTE4 (t); \ 279 OPCODE ARG1,ARG2; \ 280 li $2,TRUE; \ 281 bc1t 1f; \ 282 li $2,FALSE; \ 2831:; \ 284 j $31; \ 285 ENDFN (NAME) 286 287/* Like CMPSF, but reverse the comparison operands. */ 288 289#define REVCMPSF(NAME, OPCODE, TRUE, FALSE) \ 290STARTFN (NAME); \ 291 MOVE_SF_BYTE0 (t); \ 292 MOVE_SF_BYTE4 (t); \ 293 OPCODE ARG2,ARG1; \ 294 li $2,TRUE; \ 295 bc1t 1f; \ 296 li $2,FALSE; \ 2971:; \ 298 j $31; \ 299 ENDFN (NAME) 300 301#ifdef L_m16eqsf2 302CMPSF (__mips16_eqsf2, c.eq.s, 0, 1) 303#endif 304#ifdef L_m16nesf2 305CMPSF (__mips16_nesf2, c.eq.s, 0, 1) 306#endif 307#ifdef L_m16gtsf2 308REVCMPSF (__mips16_gtsf2, c.lt.s, 1, 0) 309#endif 310#ifdef L_m16gesf2 311REVCMPSF (__mips16_gesf2, c.le.s, 0, -1) 312#endif 313#ifdef L_m16lesf2 314CMPSF (__mips16_lesf2, c.le.s, 0, 1) 315#endif 316#ifdef L_m16ltsf2 317CMPSF (__mips16_ltsf2, c.lt.s, -1, 0) 318#endif 319#ifdef L_m16unordsf2 320CMPSF(__mips16_unordsf2, c.un.s, 1, 0) 321#endif 322 323 324/* Single-precision conversions. */ 325 326#ifdef L_m16fltsisf 327STARTFN (__mips16_floatsisf) 328 MOVE_SF_BYTE0 (t) 329 cvt.s.w RET,ARG1 330 MOVE_SF_RET (f, $31) 331 ENDFN (__mips16_floatsisf) 332#endif 333 334#ifdef L_m16fltunsisf 335STARTFN (__mips16_floatunsisf) 336 .set noreorder 337 bltz $4,1f 338 MOVE_SF_BYTE0 (t) 339 .set reorder 340 cvt.s.w RET,ARG1 341 MOVE_SF_RET (f, $31) 3421: 343 and $2,$4,1 344 srl $3,$4,1 345 or $2,$2,$3 346 mtc1 $2,RET 347 cvt.s.w RET,RET 348 add.s RET,RET,RET 349 MOVE_SF_RET (f, $31) 350 ENDFN (__mips16_floatunsisf) 351#endif 352 353#ifdef L_m16fix_truncsfsi 354STARTFN (__mips16_fix_truncsfsi) 355 MOVE_SF_BYTE0 (t) 356 trunc.w.s RET,ARG1,$4 357 MOVE_SI_RET (f, $31) 358 ENDFN (__mips16_fix_truncsfsi) 359#endif 360 361#if !defined(__mips_single_float) && !defined(__SINGLE_FLOAT) 362 363/* Double-precision math. */ 364 365/* Define a function NAME that loads two double-precision values, 366 performs FPU operation OPCODE on them, and returns the double- 367 precision result. */ 368 369#define OPDF3(NAME, OPCODE) \ 370STARTFN (NAME); \ 371 MOVE_DF_BYTE0 (t); \ 372 MOVE_DF_BYTE8 (t); \ 373 OPCODE RET,ARG1,ARG2; \ 374 MOVE_DF_RET (f, $31); \ 375 ENDFN (NAME) 376 377#ifdef L_m16adddf3 378OPDF3 (__mips16_adddf3, add.d) 379#endif 380#ifdef L_m16subdf3 381OPDF3 (__mips16_subdf3, sub.d) 382#endif 383#ifdef L_m16muldf3 384OPDF3 (__mips16_muldf3, mul.d) 385#endif 386#ifdef L_m16divdf3 387OPDF3 (__mips16_divdf3, div.d) 388#endif 389 390/* Define a function NAME that loads a double-precision value, 391 performs FPU operation OPCODE on it, and returns the double- 392 precision result. */ 393 394#define OPDF2(NAME, OPCODE) \ 395STARTFN (NAME); \ 396 MOVE_DF_BYTE0 (t); \ 397 OPCODE RET,ARG1; \ 398 MOVE_DF_RET (f, $31); \ 399 ENDFN (NAME) 400 401#ifdef L_m16negdf2 402OPDF2 (__mips16_negdf2, neg.d) 403#endif 404#ifdef L_m16absdf2 405OPDF2 (__mips16_absdf2, abs.d) 406#endif 407 408/* Conversions between single and double precision. */ 409 410#ifdef L_m16extsfdf2 411STARTFN (__mips16_extendsfdf2) 412 MOVE_SF_BYTE0 (t) 413 cvt.d.s RET,ARG1 414 MOVE_DF_RET (f, $31) 415 ENDFN (__mips16_extendsfdf2) 416#endif 417 418#ifdef L_m16trdfsf2 419STARTFN (__mips16_truncdfsf2) 420 MOVE_DF_BYTE0 (t) 421 cvt.s.d RET,ARG1 422 MOVE_SF_RET (f, $31) 423 ENDFN (__mips16_truncdfsf2) 424#endif 425 426/* Double-precision comparisons. */ 427 428/* Define a function NAME that loads two double-precision values, 429 performs floating point comparison OPCODE, and returns TRUE or 430 FALSE depending on the result. */ 431 432#define CMPDF(NAME, OPCODE, TRUE, FALSE) \ 433STARTFN (NAME); \ 434 MOVE_DF_BYTE0 (t); \ 435 MOVE_DF_BYTE8 (t); \ 436 OPCODE ARG1,ARG2; \ 437 li $2,TRUE; \ 438 bc1t 1f; \ 439 li $2,FALSE; \ 4401:; \ 441 j $31; \ 442 ENDFN (NAME) 443 444/* Like CMPDF, but reverse the comparison operands. */ 445 446#define REVCMPDF(NAME, OPCODE, TRUE, FALSE) \ 447STARTFN (NAME); \ 448 MOVE_DF_BYTE0 (t); \ 449 MOVE_DF_BYTE8 (t); \ 450 OPCODE ARG2,ARG1; \ 451 li $2,TRUE; \ 452 bc1t 1f; \ 453 li $2,FALSE; \ 4541:; \ 455 j $31; \ 456 ENDFN (NAME) 457 458#ifdef L_m16eqdf2 459CMPDF (__mips16_eqdf2, c.eq.d, 0, 1) 460#endif 461#ifdef L_m16nedf2 462CMPDF (__mips16_nedf2, c.eq.d, 0, 1) 463#endif 464#ifdef L_m16gtdf2 465REVCMPDF (__mips16_gtdf2, c.lt.d, 1, 0) 466#endif 467#ifdef L_m16gedf2 468REVCMPDF (__mips16_gedf2, c.le.d, 0, -1) 469#endif 470#ifdef L_m16ledf2 471CMPDF (__mips16_ledf2, c.le.d, 0, 1) 472#endif 473#ifdef L_m16ltdf2 474CMPDF (__mips16_ltdf2, c.lt.d, -1, 0) 475#endif 476#ifdef L_m16unorddf2 477CMPDF(__mips16_unorddf2, c.un.d, 1, 0) 478#endif 479 480/* Double-precision conversions. */ 481 482#ifdef L_m16fltsidf 483STARTFN (__mips16_floatsidf) 484 MOVE_SI_BYTE0 (t) 485 cvt.d.w RET,ARG1 486 MOVE_DF_RET (f, $31) 487 ENDFN (__mips16_floatsidf) 488#endif 489 490#ifdef L_m16fltunsidf 491STARTFN (__mips16_floatunsidf) 492 MOVE_SI_BYTE0 (t) 493 cvt.d.w RET,ARG1 494 bgez $4,1f 495 li.d ARG1, 4.294967296e+9 496 add.d RET, RET, ARG1 4971: MOVE_DF_RET (f, $31) 498 ENDFN (__mips16_floatunsidf) 499#endif 500 501#ifdef L_m16fix_truncdfsi 502STARTFN (__mips16_fix_truncdfsi) 503 MOVE_DF_BYTE0 (t) 504 trunc.w.d RET,ARG1,$4 505 MOVE_SI_RET (f, $31) 506 ENDFN (__mips16_fix_truncdfsi) 507#endif 508#endif /* !__mips_single_float */ 509 510/* We don't export stubs from libgcc_s.so and always require static 511 versions to be pulled from libgcc.a as needed because they use $2 512 and possibly $3 as arguments, diverging from the standard SysV ABI, 513 and as such would require severe pessimisation of MIPS16 PLT entries 514 just for this single special case. 515 516 For compatibility with old binaries that used safe standard MIPS PLT 517 entries and referred to these functions we still export them at 518 version GCC_4.4.0 for run-time loading only. */ 519 520#ifdef SHARED 521#define CE_STARTFN(NAME) \ 522STARTFN (NAME##_compat); \ 523 .symver NAME##_compat, NAME@GCC_4.4.0 524#define CE_ENDFN(NAME) ENDFN (NAME##_compat) 525#else 526#define CE_STARTFN(NAME) \ 527STARTFN (NAME); \ 528 .hidden NAME 529#define CE_ENDFN(NAME) ENDFN (NAME) 530#endif 531 532/* Define a function NAME that moves a return value of mode MODE from 533 FPRs to GPRs. */ 534 535#define RET_FUNCTION(NAME, MODE) \ 536CE_STARTFN (NAME); \ 537 MOVE_##MODE##_RET (t, $31); \ 538 CE_ENDFN (NAME) 539 540#ifdef L_m16retsf 541RET_FUNCTION (__mips16_ret_sf, SF) 542#endif 543 544#ifdef L_m16retsc 545RET_FUNCTION (__mips16_ret_sc, SC) 546#endif 547 548#if !defined(__mips_single_float) && !defined(__SINGLE_FLOAT) 549#ifdef L_m16retdf 550RET_FUNCTION (__mips16_ret_df, DF) 551#endif 552 553#ifdef L_m16retdc 554RET_FUNCTION (__mips16_ret_dc, DC) 555#endif 556#endif /* !__mips_single_float */ 557 558/* STUB_ARGS_X copies the arguments from GPRs to FPRs for argument 559 code X. X is calculated as ARG1 + ARG2 * 4, where ARG1 and ARG2 560 classify the first and second arguments as follows: 561 562 1: a single-precision argument 563 2: a double-precision argument 564 0: no argument, or not one of the above. */ 565 566#define STUB_ARGS_0 /* () */ 567#define STUB_ARGS_1 MOVE_SF_BYTE0 (t) /* (sf) */ 568#define STUB_ARGS_5 MOVE_SF_BYTE0 (t); MOVE_SF_BYTE4 (t) /* (sf, sf) */ 569#define STUB_ARGS_9 MOVE_SF_BYTE0 (t); MOVE_DF_BYTE8 (t) /* (sf, df) */ 570#define STUB_ARGS_2 MOVE_DF_BYTE0 (t) /* (df) */ 571#define STUB_ARGS_6 MOVE_DF_BYTE0 (t); MOVE_SF_BYTE8 (t) /* (df, sf) */ 572#define STUB_ARGS_10 MOVE_DF_BYTE0 (t); MOVE_DF_BYTE8 (t) /* (df, df) */ 573 574/* These functions are used by 16-bit code when calling via a function 575 pointer. They must copy the floating point arguments from the GPRs 576 to FPRs and then call function $2. */ 577 578#define CALL_STUB_NO_RET(NAME, CODE) \ 579CE_STARTFN (NAME); \ 580 STUB_ARGS_##CODE; \ 581 .set noreorder; \ 582 jr $2; \ 583 move $25,$2; \ 584 .set reorder; \ 585 CE_ENDFN (NAME) 586 587#ifdef L_m16stub1 588CALL_STUB_NO_RET (__mips16_call_stub_1, 1) 589#endif 590 591#ifdef L_m16stub5 592CALL_STUB_NO_RET (__mips16_call_stub_5, 5) 593#endif 594 595#if !defined(__mips_single_float) && !defined(__SINGLE_FLOAT) 596 597#ifdef L_m16stub2 598CALL_STUB_NO_RET (__mips16_call_stub_2, 2) 599#endif 600 601#ifdef L_m16stub6 602CALL_STUB_NO_RET (__mips16_call_stub_6, 6) 603#endif 604 605#ifdef L_m16stub9 606CALL_STUB_NO_RET (__mips16_call_stub_9, 9) 607#endif 608 609#ifdef L_m16stub10 610CALL_STUB_NO_RET (__mips16_call_stub_10, 10) 611#endif 612#endif /* !__mips_single_float */ 613 614/* Now we have the same set of functions, except that this time the 615 function being called returns an SFmode, SCmode, DFmode or DCmode 616 value; we need to instantiate a set for each case. The calling 617 function will arrange to preserve $18, so these functions are free 618 to use it to hold the return address. 619 620 Note that we do not know whether the function we are calling is 16 621 bit or 32 bit. However, it does not matter, because 16-bit 622 functions always return floating point values in both the gp and 623 the fp regs. It would be possible to check whether the function 624 being called is 16 bits, in which case the copy is unnecessary; 625 however, it's faster to always do the copy. */ 626 627#define CALL_STUB_RET(NAME, CODE, MODE) \ 628CE_STARTFN (NAME); \ 629 .cfi_startproc; \ 630 /* Create a fake CFA 4 bytes below the stack pointer. */ \ 631 .cfi_def_cfa 29,-4; \ 632 /* "Save" $sp in itself so we don't use the fake CFA. \ 633 This is: DW_CFA_val_expression r29, { DW_OP_reg29 }. */ \ 634 .cfi_escape 0x16,29,1,0x6d; \ 635 move $18,$31; \ 636 .cfi_register 31,18; \ 637 STUB_ARGS_##CODE; \ 638 .set noreorder; \ 639 jalr $2; \ 640 move $25,$2; \ 641 .set reorder; \ 642 MOVE_##MODE##_RET (f, $18); \ 643 .cfi_endproc; \ 644 CE_ENDFN (NAME) 645 646/* First, instantiate the single-float set. */ 647 648#ifdef L_m16stubsf0 649CALL_STUB_RET (__mips16_call_stub_sf_0, 0, SF) 650#endif 651 652#ifdef L_m16stubsf1 653CALL_STUB_RET (__mips16_call_stub_sf_1, 1, SF) 654#endif 655 656#ifdef L_m16stubsf5 657CALL_STUB_RET (__mips16_call_stub_sf_5, 5, SF) 658#endif 659 660#if !defined(__mips_single_float) && !defined(__SINGLE_FLOAT) 661#ifdef L_m16stubsf2 662CALL_STUB_RET (__mips16_call_stub_sf_2, 2, SF) 663#endif 664 665#ifdef L_m16stubsf6 666CALL_STUB_RET (__mips16_call_stub_sf_6, 6, SF) 667#endif 668 669#ifdef L_m16stubsf9 670CALL_STUB_RET (__mips16_call_stub_sf_9, 9, SF) 671#endif 672 673#ifdef L_m16stubsf10 674CALL_STUB_RET (__mips16_call_stub_sf_10, 10, SF) 675#endif 676#endif /* !__mips_single_float */ 677 678 679/* Now we have the same set of functions again, except that this time 680 the function being called returns an DFmode value. */ 681 682#if !defined(__mips_single_float) && !defined(__SINGLE_FLOAT) 683#ifdef L_m16stubdf0 684CALL_STUB_RET (__mips16_call_stub_df_0, 0, DF) 685#endif 686 687#ifdef L_m16stubdf1 688CALL_STUB_RET (__mips16_call_stub_df_1, 1, DF) 689#endif 690 691#ifdef L_m16stubdf5 692CALL_STUB_RET (__mips16_call_stub_df_5, 5, DF) 693#endif 694 695#ifdef L_m16stubdf2 696CALL_STUB_RET (__mips16_call_stub_df_2, 2, DF) 697#endif 698 699#ifdef L_m16stubdf6 700CALL_STUB_RET (__mips16_call_stub_df_6, 6, DF) 701#endif 702 703#ifdef L_m16stubdf9 704CALL_STUB_RET (__mips16_call_stub_df_9, 9, DF) 705#endif 706 707#ifdef L_m16stubdf10 708CALL_STUB_RET (__mips16_call_stub_df_10, 10, DF) 709#endif 710#endif /* !__mips_single_float */ 711 712 713/* Ho hum. Here we have the same set of functions again, this time 714 for when the function being called returns an SCmode value. */ 715 716#ifdef L_m16stubsc0 717CALL_STUB_RET (__mips16_call_stub_sc_0, 0, SC) 718#endif 719 720#ifdef L_m16stubsc1 721CALL_STUB_RET (__mips16_call_stub_sc_1, 1, SC) 722#endif 723 724#ifdef L_m16stubsc5 725CALL_STUB_RET (__mips16_call_stub_sc_5, 5, SC) 726#endif 727 728#if !defined(__mips_single_float) && !defined(__SINGLE_FLOAT) 729#ifdef L_m16stubsc2 730CALL_STUB_RET (__mips16_call_stub_sc_2, 2, SC) 731#endif 732 733#ifdef L_m16stubsc6 734CALL_STUB_RET (__mips16_call_stub_sc_6, 6, SC) 735#endif 736 737#ifdef L_m16stubsc9 738CALL_STUB_RET (__mips16_call_stub_sc_9, 9, SC) 739#endif 740 741#ifdef L_m16stubsc10 742CALL_STUB_RET (__mips16_call_stub_sc_10, 10, SC) 743#endif 744#endif /* !__mips_single_float */ 745 746 747/* Finally, another set of functions for DCmode. */ 748 749#if !defined(__mips_single_float) && !defined(__SINGLE_FLOAT) 750#ifdef L_m16stubdc0 751CALL_STUB_RET (__mips16_call_stub_dc_0, 0, DC) 752#endif 753 754#ifdef L_m16stubdc1 755CALL_STUB_RET (__mips16_call_stub_dc_1, 1, DC) 756#endif 757 758#ifdef L_m16stubdc5 759CALL_STUB_RET (__mips16_call_stub_dc_5, 5, DC) 760#endif 761 762#ifdef L_m16stubdc2 763CALL_STUB_RET (__mips16_call_stub_dc_2, 2, DC) 764#endif 765 766#ifdef L_m16stubdc6 767CALL_STUB_RET (__mips16_call_stub_dc_6, 6, DC) 768#endif 769 770#ifdef L_m16stubdc9 771CALL_STUB_RET (__mips16_call_stub_dc_9, 9, DC) 772#endif 773 774#ifdef L_m16stubdc10 775CALL_STUB_RET (__mips16_call_stub_dc_10, 10, DC) 776#endif 777#endif /* !__mips_single_float */ 778 779#endif 780#endif /* defined(__mips_micromips) || defined(__mips_soft_float) */ 781