/*
* %CopyrightBegin%
*
* Copyright Ericsson AB 2008-2016. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* %CopyrightEnd%
*/
#ifndef HIPE_RISC_GLUE_H
#define HIPE_RISC_GLUE_H
/* arch wrapper does:
* #include "hipe_${arch}_asm.h" // for NR_ARG_REGS, ${ARCH}_LEAF_WORDS
* #define NR_LEAF_WORDS ${ARCH}_LEAF_WORDS
* #define HIPE_ARCH_CALL_TO_NATIVE hipe_${arch}_call_to_native
* #define HIPE_ARCH_RETURN_TO_NATIVE hipe_${arch}_return_to_native
* #define HIPE_ARCH_TAILCALL_TO_NATIVE hipe_${arch}_tailcall_to_native
* #define HIPE_ARCH_THROW_TO_NATIVE hipe_${arch}_throw_to_native
* #include "hipe_risc_glue.h"
*/
/* Emulated code recursively calls native code.
The return address is `nbif_return', which is exported so that
tailcalls from native to emulated code can be identified. */
unsigned int HIPE_ARCH_CALL_TO_NATIVE(Process*);
AEXTERN(void,nbif_return,(void));
/* Native-mode stubs for calling emulated-mode closures. */
AEXTERN(void,nbif_ccallemu0,(void));
AEXTERN(void,nbif_ccallemu1,(void));
AEXTERN(void,nbif_ccallemu2,(void));
AEXTERN(void,nbif_ccallemu3,(void));
AEXTERN(void,nbif_ccallemu4,(void));
AEXTERN(void,nbif_ccallemu5,(void));
AEXTERN(void,nbif_ccallemu6,(void));
/* Default exception handler for native code. */
AEXTERN(void,nbif_fail,(void));
/* Emulated code returns to its native code caller. */
unsigned int HIPE_ARCH_RETURN_TO_NATIVE(Process*);
/* Emulated code tailcalls native code. */
unsigned int HIPE_ARCH_TAILCALL_TO_NATIVE(Process*);
/* Emulated code throws an exception to its native code caller. */
unsigned int HIPE_ARCH_THROW_TO_NATIVE(Process*);
static __inline__ unsigned int max(unsigned int x, unsigned int y)
{
return (x > y) ? x : y;
}
static __inline__ void hipe_arch_glue_init(void)
{
static struct hipe_sdesc_with_exnra nbif_return_sdesc = {
.exnra = (unsigned long)&nbif_fail,
.sdesc = {
.bucket = { .hvalue = (unsigned long)&nbif_return },
.fsize = 0,
.has_exnra = 1,
.arity = 0
},
};
hipe_init_sdesc_table(&nbif_return_sdesc.sdesc);
}
static __inline__ void hipe_push_risc_nra_frame(Process *p)
{
p->hipe.nsp -= 1;
p->hipe.nsp[0] = (Eterm)p->hipe.nra;
}
static __inline__ void hipe_pop_risc_nra_frame(Process *p)
{
p->hipe.nra = (void(*)(void))p->hipe.nsp[0];
p->hipe.nsp += 1;
}
/* PRE: arity <= NR_ARG_REGS */
static __inline__ void
hipe_write_risc_regs(Process *p, unsigned int arity, Eterm reg[])
{
#if NR_ARG_REGS > 0
int i;
for (i = arity; --i >= 0;)
p->def_arg_reg[i] = reg[i];
#endif
}
/* PRE: arity <= NR_ARG_REGS */
static __inline__ void
hipe_read_risc_regs(Process *p, unsigned int arity, Eterm reg[])
{
#if NR_ARG_REGS > 0
int i;
for (i = arity; --i >= 0;)
reg[i] = p->def_arg_reg[i];
#endif
}
static __inline__ void
hipe_push_risc_params(Process *p, unsigned int arity, Eterm reg[])
{
unsigned int i;
i = arity;
if (i > NR_ARG_REGS) {
Eterm *nsp = p->hipe.nsp;
i = NR_ARG_REGS;
do {
*--nsp = reg[i++];
} while (i < arity);
p->hipe.nsp = nsp;
i = NR_ARG_REGS;
}
/* INV: i <= NR_ARG_REGS */
hipe_write_risc_regs(p, i, reg);
}
static __inline__ void
hipe_pop_risc_params(Process *p, unsigned int arity, Eterm reg[])
{
unsigned int i;
i = arity;
if (i > NR_ARG_REGS) {
Eterm *nsp = p->hipe.nsp;
do {
reg[--i] = *nsp++;
} while (i > NR_ARG_REGS);
p->hipe.nsp = nsp;
/* INV: i == NR_ARG_REGS */
}
/* INV: i <= NR_ARG_REGS */
hipe_read_risc_regs(p, i, reg);
}
/* BEAM recursively calls native code. */
static __inline__ unsigned int
hipe_call_to_native(Process *p, unsigned int arity, Eterm reg[])
{
int nstkargs;
if ((nstkargs = arity - NR_ARG_REGS) < 0)
nstkargs = 0;
hipe_check_nstack(p, max(nstkargs + 1, NR_LEAF_WORDS));
hipe_push_risc_nra_frame(p); /* needs 1 word */
hipe_push_risc_params(p, arity, reg); /* needs nstkargs words */
return HIPE_ARCH_CALL_TO_NATIVE(p);
}
/* Native called BEAM, which now tailcalls native. */
static __inline__ unsigned int
hipe_tailcall_to_native(Process *p, unsigned int arity, Eterm reg[])
{
int nstkargs;
if ((nstkargs = arity - NR_ARG_REGS) < 0)
nstkargs = 0;
hipe_check_nstack(p, max(nstkargs, NR_LEAF_WORDS));
hipe_push_risc_params(p, arity, reg); /* needs nstkargs words */
return HIPE_ARCH_TAILCALL_TO_NATIVE(p);
}
/* BEAM called native, which has returned. Clean up. */
static __inline__ void hipe_return_from_native(Process *p)
{
hipe_pop_risc_nra_frame(p);
}
/* BEAM called native, which has thrown an exception. Clean up. */
static __inline__ void hipe_throw_from_native(Process *p)
{
hipe_pop_risc_nra_frame(p);
}
/* BEAM called native, which now calls BEAM.
Move the parameters to reg[].
Return zero if this is a tailcall, non-zero if the call is recursive.
If tailcall, also clean up native stub continuation. */
static __inline__ int
hipe_call_from_native_is_recursive(Process *p, Eterm reg[])
{
hipe_pop_risc_params(p, p->arity, reg);
if (p->hipe.nra != (void(*)(void))&nbif_return)
return 1;
hipe_pop_risc_nra_frame(p);
return 0;
}
/* BEAM called native, which called BIF that returned trap
* Discard bif parameters.
* If tailcall, also clean up native stub continuation. */
static __inline__ int
hipe_trap_from_native_is_recursive(Process *p)
{
if (p->hipe.narity > NR_ARG_REGS) {
p->hipe.nsp += (p->hipe.narity - NR_ARG_REGS);
}
if (p->hipe.nra != (void(*)(void))&nbif_return)
return 1;
hipe_pop_risc_nra_frame(p);
return 0;
}
/* Native called BIF. Is it a recursive call?
i.e should we return back to native when BIF is done? */
static __inline__ int
hipe_bifcall_from_native_is_recursive(Process *p)
{
return (p->hipe.nra != (void(*)(void))&nbif_return);
}
/* Native makes a call which needs to unload the parameters.
This differs from hipe_call_from_native_is_recursive() in
that it doesn't check for or pop the BEAM-calls-native frame.
It's currently only used in the implementation of apply. */
static __inline__ void
hipe_pop_params(Process *p, unsigned int arity, Eterm reg[])
{
hipe_pop_risc_params(p, arity, reg);
}
/* Native called BEAM, which now returns back to native. */
static __inline__ unsigned int hipe_return_to_native(Process *p)
{
return HIPE_ARCH_RETURN_TO_NATIVE(p);
}
/* Native called BEAM, which now throws an exception back to native. */
static __inline__ unsigned int hipe_throw_to_native(Process *p)
{
return HIPE_ARCH_THROW_TO_NATIVE(p);
}
/* Return the address of a stub switching a native closure call to BEAM. */
static __inline__ const void *hipe_closure_stub_address(unsigned int arity)
{
#if NR_ARG_REGS == 0
return &nbif_ccallemu0;
#else /* > 0 */
switch (arity) {
case 0: return &nbif_ccallemu0;
#if NR_ARG_REGS == 1
default: return &nbif_ccallemu1;
#else /* > 1 */
case 1: return &nbif_ccallemu1;
#if NR_ARG_REGS == 2
default: return &nbif_ccallemu2;
#else /* > 2 */
case 2: return &nbif_ccallemu2;
#if NR_ARG_REGS == 3
default: return &nbif_ccallemu3;
#else /* > 3 */
case 3: return &nbif_ccallemu3;
#if NR_ARG_REGS == 4
default: return &nbif_ccallemu4;
#else /* > 4 */
case 4: return &nbif_ccallemu4;
#if NR_ARG_REGS == 5
default: return &nbif_ccallemu5;
#else /* > 5 */
case 5: return &nbif_ccallemu5;
#if NR_ARG_REGS == 6
default: return &nbif_ccallemu6;
#else
#error "NR_ARG_REGS > 6 NOT YET IMPLEMENTED"
#endif /* > 6 */
#endif /* > 5 */
#endif /* > 4 */
#endif /* > 3 */
#endif /* > 2 */
#endif /* > 1 */
}
#endif /* > 0 */
}
#endif /* HIPE_RISC_GLUE_H */