blob: e0765b16b0b0917f8b4eab85d67c12a23b084f5c [file] [log] [blame]
Edward O'Callaghan2bf62722009-08-05 04:02:56 +00001/* ===----- trampoline_setup.c - Implement __trampoline_setup -------------===
2 *
3 * The LLVM Compiler Infrastructure
4 *
Howard Hinnant9ad441f2010-11-16 22:13:33 +00005 * This file is dual licensed under the MIT and the University of Illinois Open
6 * Source Licenses. See LICENSE.TXT for details.
Edward O'Callaghan2bf62722009-08-05 04:02:56 +00007 *
8 * ===----------------------------------------------------------------------===
9 */
Daniel Dunbarb3a69012009-06-26 16:47:03 +000010
Nick Kledzikd0d99132010-04-15 20:37:56 +000011#include "int_lib.h"
12
Daniel Dunbarb3a69012009-06-26 16:47:03 +000013extern void __clear_cache(void* start, void* end);
14
Edward O'Callaghan2bf62722009-08-05 04:02:56 +000015/*
16 * The ppc compiler generates calls to __trampoline_setup() when creating
17 * trampoline functions on the stack for use with nested functions.
18 * This function creates a custom 40-byte trampoline function on the stack
19 * which loads r11 with a pointer to the outer function's locals
20 * and then jumps to the target nested function.
21 */
22
Bob Wilsona98b7cb2012-01-03 18:59:25 +000023#if __ppc__ && !defined(__powerpc64__)
Daniel Dunbarb3a69012009-06-26 16:47:03 +000024void __trampoline_setup(uint32_t* trampOnStack, int trampSizeAllocated,
25 const void* realFunc, void* localsPtr)
26{
Edward O'Callaghan2bf62722009-08-05 04:02:56 +000027 /* should never happen, but if compiler did not allocate */
28 /* enough space on stack for the trampoline, abort */
Daniel Dunbarb3a69012009-06-26 16:47:03 +000029 if ( trampSizeAllocated < 40 )
Daniel Dunbar48f46ac2010-03-31 17:00:45 +000030 compilerrt_abort();
Daniel Dunbarb3a69012009-06-26 16:47:03 +000031
Edward O'Callaghan2bf62722009-08-05 04:02:56 +000032 /* create trampoline */
33 trampOnStack[0] = 0x7c0802a6; /* mflr r0 */
34 trampOnStack[1] = 0x4800000d; /* bl Lbase */
Daniel Dunbarb3a69012009-06-26 16:47:03 +000035 trampOnStack[2] = (uint32_t)realFunc;
36 trampOnStack[3] = (uint32_t)localsPtr;
Edward O'Callaghan2bf62722009-08-05 04:02:56 +000037 trampOnStack[4] = 0x7d6802a6; /* Lbase: mflr r11 */
38 trampOnStack[5] = 0x818b0000; /* lwz r12,0(r11) */
39 trampOnStack[6] = 0x7c0803a6; /* mtlr r0 */
40 trampOnStack[7] = 0x7d8903a6; /* mtctr r12 */
41 trampOnStack[8] = 0x816b0004; /* lwz r11,4(r11) */
42 trampOnStack[9] = 0x4e800420; /* bctr */
Daniel Dunbarb3a69012009-06-26 16:47:03 +000043
Edward O'Callaghan2bf62722009-08-05 04:02:56 +000044 /* clear instruction cache */
Daniel Dunbarb3a69012009-06-26 16:47:03 +000045 __clear_cache(trampOnStack, &trampOnStack[10]);
46}
Bob Wilsona98b7cb2012-01-03 18:59:25 +000047#endif /* __ppc__ && !defined(__powerpc64__) */