blob: ff0cc21305fc2faf2aa4293e44ac5a70fb2e7b4f [file] [log] [blame]
Damien George04b91472014-05-03 23:27:38 +01001/*
2 * This file is part of the Micro Python project, http://micropython.org/
3 *
4 * The MIT License (MIT)
5 *
6 * Copyright (c) 2013, 2014 Damien P. George
7 *
8 * Permission is hereby granted, free of charge, to any person obtaining a copy
9 * of this software and associated documentation files (the "Software"), to deal
10 * in the Software without restriction, including without limitation the rights
11 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
12 * copies of the Software, and to permit persons to whom the Software is
13 * furnished to do so, subject to the following conditions:
14 *
15 * The above copyright notice and this permission notice shall be included in
16 * all copies or substantial portions of the Software.
17 *
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
21 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
23 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
24 * THE SOFTWARE.
25 */
26
Paul Sokolovskyb1ce37d2014-04-30 04:19:20 +030027#if defined(__i386__) && !MICROPY_NLR_SETJMP
Damien George133b0832014-09-26 13:07:26 +000028
29// We only need the functions here if we are on x86, and we are not
30// using setjmp/longjmp.
31//
32// For reference, x86 callee save regs are:
33// ebx, esi, edi, ebp, esp, eip
Damience89a212013-10-15 22:25:17 +010034
Damien Georgeb4b10fd2015-01-01 23:30:53 +000035// the offset of nlr_top within mp_state_ctx_t
36#define NLR_TOP_OFFSET (2 * 4)
37
Igor Gatisf5c554d2015-11-19 13:53:37 -030038#if defined(_WIN32) || defined(__CYGWIN__)
stijnbf1570c2015-11-20 15:18:42 +010039#define NLR_OS_WINDOWS
Jan Pochylaffb04a52016-06-02 19:10:39 +020040#endif
41
42#if defined(__APPLE__) && defined(__MACH__)
43#define NLR_OS_MAC
44#endif
45
46#if defined(NLR_OS_WINDOWS) || defined(NLR_OS_MAC)
Damien Georgeb4b10fd2015-01-01 23:30:53 +000047#define NLR_TOP (_mp_state_ctx + NLR_TOP_OFFSET)
Jan Pochylaffb04a52016-06-02 19:10:39 +020048#define MP_THREAD_GET_STATE _mp_thread_get_state
Damien George872a8292015-01-01 22:03:44 +000049#else
Damien Georgeb4b10fd2015-01-01 23:30:53 +000050#define NLR_TOP (mp_state_ctx + NLR_TOP_OFFSET)
Jan Pochylaffb04a52016-06-02 19:10:39 +020051#define MP_THREAD_GET_STATE mp_thread_get_state
Damien George872a8292015-01-01 22:03:44 +000052#endif
53
Damien George27cc0772016-04-22 22:52:33 +000054// offset of nlr_top within mp_state_thread_t structure
55#define NLR_TOP_TH_OFF (0)
56
Damience89a212013-10-15 22:25:17 +010057 .file "nlr.s"
58 .text
59
Damien George133b0832014-09-26 13:07:26 +000060/**************************************/
61// mp_uint_t nlr_push(4(%esp)=nlr_buf_t *nlr)
62
stijnbf1570c2015-11-20 15:18:42 +010063#if defined(NLR_OS_WINDOWS)
Markus Siemens242856c2014-01-28 19:52:04 +010064 .globl _nlr_push
65 .def _nlr_push; .scl 2; .type 32; .endef
66_nlr_push:
Jan Pochylaffb04a52016-06-02 19:10:39 +020067#elif defined(NLR_OS_MAC)
68 .globl _nlr_push
69_nlr_push:
Markus Siemens242856c2014-01-28 19:52:04 +010070#else
Damience89a212013-10-15 22:25:17 +010071 .globl nlr_push
72 .type nlr_push, @function
73nlr_push:
Markus Siemens19ccc6b2014-01-27 22:53:28 +010074#endif
Damience89a212013-10-15 22:25:17 +010075 mov 4(%esp), %edx # load nlr_buf
76 mov (%esp), %eax # load return %ip
77 mov %eax, 8(%edx) # store %ip into nlr_buf+8
78 mov %ebp, 12(%edx) # store %bp into nlr_buf+12
79 mov %esp, 16(%edx) # store %sp into nlr_buf+16
80 mov %ebx, 20(%edx) # store %bx into nlr_buf+20
81 mov %edi, 24(%edx) # store %di into nlr_buf
82 mov %esi, 28(%edx) # store %si into nlr_buf
Damien George27cc0772016-04-22 22:52:33 +000083
84#if !MICROPY_PY_THREAD
Damien George872a8292015-01-01 22:03:44 +000085 mov NLR_TOP, %eax # load nlr_top
Damience89a212013-10-15 22:25:17 +010086 mov %eax, (%edx) # store it
Damien George872a8292015-01-01 22:03:44 +000087 mov %edx, NLR_TOP # stor new nlr_buf (to make linked list)
Damien George27cc0772016-04-22 22:52:33 +000088#else
89 // to check: stack is aligned to 16-byte boundary before this call
Jan Pochylaffb04a52016-06-02 19:10:39 +020090 call MP_THREAD_GET_STATE # get mp_state_thread ptr into eax
Damien George27cc0772016-04-22 22:52:33 +000091 mov 4(%esp), %edx # load nlr_buf argument into edx (edx clobbered by call)
92 mov NLR_TOP_TH_OFF(%eax), %ecx # get thread.nlr_top (last nlr_buf)
93 mov %ecx, (%edx) # store it
94 mov %edx, NLR_TOP_TH_OFF(%eax) # store new nlr_buf (to make linked list)
95#endif
96
Damience89a212013-10-15 22:25:17 +010097 xor %eax, %eax # return 0, normal return
98 ret # return
Jan Pochylaffb04a52016-06-02 19:10:39 +020099#if !defined(NLR_OS_WINDOWS) && !defined(NLR_OS_MAC)
Damience89a212013-10-15 22:25:17 +0100100 .size nlr_push, .-nlr_push
Markus Siemens19ccc6b2014-01-27 22:53:28 +0100101#endif
Damience89a212013-10-15 22:25:17 +0100102
Damien George133b0832014-09-26 13:07:26 +0000103/**************************************/
104// void nlr_pop()
105
stijnbf1570c2015-11-20 15:18:42 +0100106#if defined(NLR_OS_WINDOWS)
Markus Siemens242856c2014-01-28 19:52:04 +0100107 .globl _nlr_pop
108 .def _nlr_pop; .scl 2; .type 32; .endef
109_nlr_pop:
Jan Pochylaffb04a52016-06-02 19:10:39 +0200110#elif defined(NLR_OS_MAC)
111 .globl _nlr_pop
112_nlr_pop:
Markus Siemens242856c2014-01-28 19:52:04 +0100113#else
Damience89a212013-10-15 22:25:17 +0100114 .globl nlr_pop
115 .type nlr_pop, @function
116nlr_pop:
Markus Siemens19ccc6b2014-01-27 22:53:28 +0100117#endif
Damien George27cc0772016-04-22 22:52:33 +0000118
119#if !MICROPY_PY_THREAD
Damien George872a8292015-01-01 22:03:44 +0000120 mov NLR_TOP, %eax # load nlr_top
Damience89a212013-10-15 22:25:17 +0100121 mov (%eax), %eax # load prev nlr_buf
Damien George872a8292015-01-01 22:03:44 +0000122 mov %eax, NLR_TOP # store nlr_top (to unlink list)
Damien George27cc0772016-04-22 22:52:33 +0000123#else
Jan Pochylaffb04a52016-06-02 19:10:39 +0200124 call MP_THREAD_GET_STATE # get mp_state_thread ptr into eax
Damien George27cc0772016-04-22 22:52:33 +0000125 mov NLR_TOP_TH_OFF(%eax), %ecx # get thread.nlr_top (last nlr_buf)
126 mov (%ecx), %ecx # load prev nlr_buf
127 mov %ecx, NLR_TOP_TH_OFF(%eax) # store prev nlr_buf (to unlink list)
128#endif
129
Damience89a212013-10-15 22:25:17 +0100130 ret # return
Jan Pochylaffb04a52016-06-02 19:10:39 +0200131#if !defined(NLR_OS_WINDOWS) && !defined(NLR_OS_MAC)
Damience89a212013-10-15 22:25:17 +0100132 .size nlr_pop, .-nlr_pop
Markus Siemens19ccc6b2014-01-27 22:53:28 +0100133#endif
Damience89a212013-10-15 22:25:17 +0100134
Damien George133b0832014-09-26 13:07:26 +0000135/**************************************/
136// void nlr_jump(4(%esp)=mp_uint_t val)
137
stijnbf1570c2015-11-20 15:18:42 +0100138#if defined(NLR_OS_WINDOWS)
Markus Siemens242856c2014-01-28 19:52:04 +0100139 .globl _nlr_jump
140 .def _nlr_jump; .scl 2; .type 32; .endef
141_nlr_jump:
Jan Pochylaffb04a52016-06-02 19:10:39 +0200142#elif defined(NLR_OS_MAC)
143 .globl _nlr_jump
144_nlr_jump:
Markus Siemens242856c2014-01-28 19:52:04 +0100145#else
Damience89a212013-10-15 22:25:17 +0100146 .globl nlr_jump
147 .type nlr_jump, @function
148nlr_jump:
Markus Siemens19ccc6b2014-01-27 22:53:28 +0100149#endif
Damien George27cc0772016-04-22 22:52:33 +0000150
151#if !MICROPY_PY_THREAD
Damien George872a8292015-01-01 22:03:44 +0000152 mov NLR_TOP, %edx # load nlr_top
Damien George26cf55a2014-04-08 14:08:14 +0000153 test %edx, %edx # check for nlr_top being NULL
Jan Pochylaffb04a52016-06-02 19:10:39 +0200154#if defined(NLR_OS_WINDOWS) || defined(NLR_OS_MAC)
Paul Sokolovsky41809a12014-04-20 22:14:58 +0300155 je _nlr_jump_fail # fail if nlr_top is NULL
156#else
Damien George26cf55a2014-04-08 14:08:14 +0000157 je nlr_jump_fail # fail if nlr_top is NULL
Paul Sokolovsky41809a12014-04-20 22:14:58 +0300158#endif
Damience89a212013-10-15 22:25:17 +0100159 mov 4(%esp), %eax # load return value
160 mov %eax, 4(%edx) # store return value
161 mov (%edx), %eax # load prev nlr_top
Damien George872a8292015-01-01 22:03:44 +0000162 mov %eax, NLR_TOP # store nlr_top (to unlink list)
Damien George27cc0772016-04-22 22:52:33 +0000163#else
Jan Pochylaffb04a52016-06-02 19:10:39 +0200164 call MP_THREAD_GET_STATE # get mp_state_thread ptr into eax
Damien George27cc0772016-04-22 22:52:33 +0000165 mov NLR_TOP_TH_OFF(%eax), %edx # get thread.nlr_top (last nlr_buf)
166 test %edx, %edx # check for nlr_top being NULL
Jan Pochylaffb04a52016-06-02 19:10:39 +0200167#if defined(NLR_OS_WINDOWS) || defined(NLR_OS_MAC)
Damien George27cc0772016-04-22 22:52:33 +0000168 je _nlr_jump_fail # fail if nlr_top is NULL
169#else
170 je nlr_jump_fail # fail if nlr_top is NULL
171#endif
172 mov 4(%esp), %ecx # load return value
173 mov %ecx, 4(%edx) # store return value
174 mov (%edx), %ecx # load prev nlr_top
175 mov %ecx, NLR_TOP_TH_OFF(%eax) # store nlr_top (to unlink list)
176#endif
177
Damience89a212013-10-15 22:25:17 +0100178 mov 28(%edx), %esi # load saved %si
179 mov 24(%edx), %edi # load saved %di
180 mov 20(%edx), %ebx # load saved %bx
181 mov 16(%edx), %esp # load saved %sp
182 mov 12(%edx), %ebp # load saved %bp
183 mov 8(%edx), %eax # load saved %ip
184 mov %eax, (%esp) # store saved %ip to stack
185 xor %eax, %eax # clear return register
186 inc %al # increase to make 1, non-local return
187 ret # return
Jan Pochylaffb04a52016-06-02 19:10:39 +0200188#if !defined(NLR_OS_WINDOWS) && !defined(NLR_OS_MAC)
Damience89a212013-10-15 22:25:17 +0100189 .size nlr_jump, .-nlr_jump
Markus Siemens19ccc6b2014-01-27 22:53:28 +0100190#endif
Damience89a212013-10-15 22:25:17 +0100191
Damien George133b0832014-09-26 13:07:26 +0000192#endif // defined(__i386__) && !MICROPY_NLR_SETJMP
Dave Hylandsaee74a12017-01-30 20:35:52 -0800193#if defined(linux)
Dave Hylandsaa34c552017-02-07 14:22:55 -0800194 .section .note.GNU-stack,"",%progbits
Dave Hylandsaee74a12017-01-30 20:35:52 -0800195#endif