jdk-24/test/jdk/java/foreign/libVarArgs.c
2022-05-24 15:03:53 +00:00

236 lines
6.3 KiB
C

/*
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*
*/
#include <stdarg.h>
#include <stdlib.h>
#include "shared.h"
typedef void (*writeback_t)(int,void*);
typedef struct {
writeback_t writeback;
int* argids;
} call_info;
#define CASE(num, type) case num: { \
type x = va_arg(a_list, type); \
writeback(i, &x); \
} break;
enum NativeType {
T_INT,
T_FLOAT,
T_DOUBLE,
T_POINTER,
T_S_I,
T_S_F,
T_S_D,
T_S_P,
T_S_II,
T_S_IF,
T_S_ID,
T_S_IP,
T_S_FI,
T_S_FF,
T_S_FD,
T_S_FP,
T_S_DI,
T_S_DF,
T_S_DD,
T_S_DP,
T_S_PI,
T_S_PF,
T_S_PD,
T_S_PP,
T_S_III,
T_S_IIF,
T_S_IID,
T_S_IIP,
T_S_IFI,
T_S_IFF,
T_S_IFD,
T_S_IFP,
T_S_IDI,
T_S_IDF,
T_S_IDD,
T_S_IDP,
T_S_IPI,
T_S_IPF,
T_S_IPD,
T_S_IPP,
T_S_FII,
T_S_FIF,
T_S_FID,
T_S_FIP,
T_S_FFI,
T_S_FFF,
T_S_FFD,
T_S_FFP,
T_S_FDI,
T_S_FDF,
T_S_FDD,
T_S_FDP,
T_S_FPI,
T_S_FPF,
T_S_FPD,
T_S_FPP,
T_S_DII,
T_S_DIF,
T_S_DID,
T_S_DIP,
T_S_DFI,
T_S_DFF,
T_S_DFD,
T_S_DFP,
T_S_DDI,
T_S_DDF,
T_S_DDD,
T_S_DDP,
T_S_DPI,
T_S_DPF,
T_S_DPD,
T_S_DPP,
T_S_PII,
T_S_PIF,
T_S_PID,
T_S_PIP,
T_S_PFI,
T_S_PFF,
T_S_PFD,
T_S_PFP,
T_S_PDI,
T_S_PDF,
T_S_PDD,
T_S_PDP,
T_S_PPI,
T_S_PPF,
T_S_PPD,
T_S_PPP,
};
// need to pass `num` separately as last argument preceding varargs according to spec (and for MSVC)
EXPORT void varargs(call_info* info, int num, ...) {
va_list a_list;
va_start(a_list, num);
writeback_t writeback = info->writeback;
for (int i = 0; i < num; i++) {
int id = info->argids[i];
switch (id) {
CASE(T_INT, int)
CASE(T_FLOAT, double) // vararg float is promoted to double per C spec
CASE(T_DOUBLE, double)
CASE(T_POINTER, void*)
CASE(T_S_I, struct S_I)
CASE(T_S_F, struct S_F)
CASE(T_S_D, struct S_D)
CASE(T_S_P, struct S_P)
CASE(T_S_II, struct S_II)
CASE(T_S_IF, struct S_IF)
CASE(T_S_ID, struct S_ID)
CASE(T_S_IP, struct S_IP)
CASE(T_S_FI, struct S_FI)
CASE(T_S_FF, struct S_FF)
CASE(T_S_FD, struct S_FD)
CASE(T_S_FP, struct S_FP)
CASE(T_S_DI, struct S_DI)
CASE(T_S_DF, struct S_DF)
CASE(T_S_DD, struct S_DD)
CASE(T_S_DP, struct S_DP)
CASE(T_S_PI, struct S_PI)
CASE(T_S_PF, struct S_PF)
CASE(T_S_PD, struct S_PD)
CASE(T_S_PP, struct S_PP)
CASE(T_S_III, struct S_III)
CASE(T_S_IIF, struct S_IIF)
CASE(T_S_IID, struct S_IID)
CASE(T_S_IIP, struct S_IIP)
CASE(T_S_IFI, struct S_IFI)
CASE(T_S_IFF, struct S_IFF)
CASE(T_S_IFD, struct S_IFD)
CASE(T_S_IFP, struct S_IFP)
CASE(T_S_IDI, struct S_IDI)
CASE(T_S_IDF, struct S_IDF)
CASE(T_S_IDD, struct S_IDD)
CASE(T_S_IDP, struct S_IDP)
CASE(T_S_IPI, struct S_IPI)
CASE(T_S_IPF, struct S_IPF)
CASE(T_S_IPD, struct S_IPD)
CASE(T_S_IPP, struct S_IPP)
CASE(T_S_FII, struct S_FII)
CASE(T_S_FIF, struct S_FIF)
CASE(T_S_FID, struct S_FID)
CASE(T_S_FIP, struct S_FIP)
CASE(T_S_FFI, struct S_FFI)
CASE(T_S_FFF, struct S_FFF)
CASE(T_S_FFD, struct S_FFD)
CASE(T_S_FFP, struct S_FFP)
CASE(T_S_FDI, struct S_FDI)
CASE(T_S_FDF, struct S_FDF)
CASE(T_S_FDD, struct S_FDD)
CASE(T_S_FDP, struct S_FDP)
CASE(T_S_FPI, struct S_FPI)
CASE(T_S_FPF, struct S_FPF)
CASE(T_S_FPD, struct S_FPD)
CASE(T_S_FPP, struct S_FPP)
CASE(T_S_DII, struct S_DII)
CASE(T_S_DIF, struct S_DIF)
CASE(T_S_DID, struct S_DID)
CASE(T_S_DIP, struct S_DIP)
CASE(T_S_DFI, struct S_DFI)
CASE(T_S_DFF, struct S_DFF)
CASE(T_S_DFD, struct S_DFD)
CASE(T_S_DFP, struct S_DFP)
CASE(T_S_DDI, struct S_DDI)
CASE(T_S_DDF, struct S_DDF)
CASE(T_S_DDD, struct S_DDD)
CASE(T_S_DDP, struct S_DDP)
CASE(T_S_DPI, struct S_DPI)
CASE(T_S_DPF, struct S_DPF)
CASE(T_S_DPD, struct S_DPD)
CASE(T_S_DPP, struct S_DPP)
CASE(T_S_PII, struct S_PII)
CASE(T_S_PIF, struct S_PIF)
CASE(T_S_PID, struct S_PID)
CASE(T_S_PIP, struct S_PIP)
CASE(T_S_PFI, struct S_PFI)
CASE(T_S_PFF, struct S_PFF)
CASE(T_S_PFD, struct S_PFD)
CASE(T_S_PFP, struct S_PFP)
CASE(T_S_PDI, struct S_PDI)
CASE(T_S_PDF, struct S_PDF)
CASE(T_S_PDD, struct S_PDD)
CASE(T_S_PDP, struct S_PDP)
CASE(T_S_PPI, struct S_PPI)
CASE(T_S_PPF, struct S_PPF)
CASE(T_S_PPD, struct S_PPD)
CASE(T_S_PPP, struct S_PPP)
default: exit(-1); // invalid id
}
}
va_end(a_list);
}