blob: dd4d4d55463a5f8ebc6d743e5c66fdd0571484ef [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
Damien Georgee1e359f2015-02-07 17:24:10 +000027#include "py/mpstate.h"
Damien George51dfcb42015-01-01 20:27:54 +000028#include "py/nlr.h"
29#include "py/builtin.h"
30#include "py/objlist.h"
31#include "py/objtuple.h"
32#include "py/objstr.h"
33#include "py/objint.h"
Damien George51dfcb42015-01-01 20:27:54 +000034#include "py/stream.h"
Paul Sokolovsky5500cde2014-04-13 06:43:18 +030035
Damien Georgeee3fd462014-05-24 23:03:12 +010036#if MICROPY_PY_SYS
Paul Sokolovsky5500cde2014-04-13 06:43:18 +030037
Damien George95f53462015-04-22 17:38:05 +010038#include "genhdr/mpversion.h"
Damien Georgec3184ae2015-04-21 14:45:04 +000039
Damien George30dd23a2014-08-10 17:50:28 +010040/// \module sys - system specific functions
41
Damien Georgecd342072015-01-12 22:30:49 +000042// defined per port; type of these is irrelevant, just need pointer
stijne50cff62015-04-09 12:27:15 +020043extern struct _mp_dummy_t mp_sys_stdin_obj;
44extern struct _mp_dummy_t mp_sys_stdout_obj;
45extern struct _mp_dummy_t mp_sys_stderr_obj;
Damien Georgecd342072015-01-12 22:30:49 +000046
Damien George5ae5ec92015-04-11 12:01:39 +010047#if MICROPY_PY_IO
Damien George999cedb2015-11-27 17:01:44 +000048const mp_print_t mp_sys_stdout_print = {&mp_sys_stdout_obj, mp_stream_write_adaptor};
Damien George5ae5ec92015-04-11 12:01:39 +010049#endif
50
Damien George30dd23a2014-08-10 17:50:28 +010051/// \constant version - Python language version that this implementation conforms to, as a string
52STATIC const MP_DEFINE_STR_OBJ(version_obj, "3.4.0");
53
Damien George4ef26c12014-08-10 17:53:43 +010054/// \constant version_info - Python language version that this implementation conforms to, as a tuple of ints
Paul Sokolovskybaaaf652014-04-13 09:46:58 +030055#define I(n) MP_OBJ_NEW_SMALL_INT(n)
56// TODO: CPython is now at 5-element array, but save 2 els so far...
Paul Sokolovskybbae42d2014-04-14 01:46:45 +030057STATIC const mp_obj_tuple_t mp_sys_version_info_obj = {{&mp_type_tuple}, 3, {I(3), I(4), I(0)}};
Damien Georgec3184ae2015-04-21 14:45:04 +000058
59// sys.implementation object
60// this holds the MicroPython version
61STATIC const mp_obj_tuple_t mp_sys_implementation_version_info_obj = {
62 {&mp_type_tuple},
63 3,
64 { I(MICROPY_VERSION_MAJOR), I(MICROPY_VERSION_MINOR), I(MICROPY_VERSION_MICRO) }
65};
66#if MICROPY_PY_ATTRTUPLE
67STATIC const qstr impl_fields[] = { MP_QSTR_name, MP_QSTR_version };
68STATIC MP_DEFINE_ATTRTUPLE(
69 mp_sys_implementation_obj,
70 impl_fields,
71 2,
Damien Georgecbf76742015-11-27 13:38:15 +000072 MP_ROM_QSTR(MP_QSTR_micropython),
73 MP_ROM_PTR(&mp_sys_implementation_version_info_obj)
Damien Georgec3184ae2015-04-21 14:45:04 +000074);
75#else
76STATIC const mp_obj_tuple_t mp_sys_implementation_obj = {
77 {&mp_type_tuple},
78 2,
79 {
80 MP_OBJ_NEW_QSTR(MP_QSTR_micropython),
81 (mp_obj_t)&mp_sys_implementation_version_info_obj,
82 }
83};
84#endif
85
Paul Sokolovskybaaaf652014-04-13 09:46:58 +030086#undef I
Damien George30dd23a2014-08-10 17:50:28 +010087
Paul Sokolovskyb9b93542014-06-07 23:40:04 +030088#ifdef MICROPY_PY_SYS_PLATFORM
Damien George30dd23a2014-08-10 17:50:28 +010089/// \constant platform - the platform that Micro Python is running on
Paul Sokolovskyb9b93542014-06-07 23:40:04 +030090STATIC const MP_DEFINE_STR_OBJ(platform_obj, MICROPY_PY_SYS_PLATFORM);
91#endif
Paul Sokolovsky5500cde2014-04-13 06:43:18 +030092
Damien Georgeb92cbe62014-09-15 15:53:09 +010093/// \function exit([retval])
94/// Raise a `SystemExit` exception. If an argument is given, it is the
95/// value given to `SystemExit`.
96STATIC mp_obj_t mp_sys_exit(mp_uint_t n_args, const mp_obj_t *args) {
97 mp_obj_t exc;
98 if (n_args == 0) {
99 exc = mp_obj_new_exception(&mp_type_SystemExit);
100 } else {
101 exc = mp_obj_new_exception_arg1(&mp_type_SystemExit, args[0]);
102 }
103 nlr_raise(exc);
104}
105MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(mp_sys_exit_obj, 0, 1, mp_sys_exit);
106
Paul Sokolovsky46c3ab22014-12-06 14:29:09 +0200107STATIC mp_obj_t mp_sys_print_exception(mp_uint_t n_args, const mp_obj_t *args) {
108 #if MICROPY_PY_IO
Damien George999cedb2015-11-27 17:01:44 +0000109 void *stream_obj = &mp_sys_stdout_obj;
Paul Sokolovsky46c3ab22014-12-06 14:29:09 +0200110 if (n_args > 1) {
Damien George999cedb2015-11-27 17:01:44 +0000111 stream_obj = MP_OBJ_TO_PTR(args[1]); // XXX may fail
Paul Sokolovsky46c3ab22014-12-06 14:29:09 +0200112 }
113
Damien George999cedb2015-11-27 17:01:44 +0000114 mp_print_t print = {stream_obj, mp_stream_write_adaptor};
Damien George7f9d1d62015-04-09 23:56:15 +0100115 mp_obj_print_exception(&print, args[0]);
Paul Sokolovsky46c3ab22014-12-06 14:29:09 +0200116 #else
Damien George3a2171e2015-09-04 16:53:46 +0100117 (void)n_args;
Damien George7f9d1d62015-04-09 23:56:15 +0100118 mp_obj_print_exception(&mp_plat_print, args[0]);
Paul Sokolovsky46c3ab22014-12-06 14:29:09 +0200119 #endif
120
121 return mp_const_none;
122}
123MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(mp_sys_print_exception_obj, 1, 2, mp_sys_print_exception);
124
Paul Sokolovsky8b85d142015-04-25 03:17:41 +0300125#if MICROPY_PY_SYS_EXC_INFO
126STATIC mp_obj_t mp_sys_exc_info(void) {
Damien George999cedb2015-11-27 17:01:44 +0000127 mp_obj_t cur_exc = MP_OBJ_FROM_PTR(MP_STATE_VM(cur_exception));
128 mp_obj_tuple_t *t = MP_OBJ_TO_PTR(mp_obj_new_tuple(3, NULL));
Paul Sokolovsky8b85d142015-04-25 03:17:41 +0300129
130 if (cur_exc == MP_OBJ_NULL) {
131 t->items[0] = mp_const_none;
132 t->items[1] = mp_const_none;
133 t->items[2] = mp_const_none;
Damien George999cedb2015-11-27 17:01:44 +0000134 return MP_OBJ_FROM_PTR(t);
Paul Sokolovsky8b85d142015-04-25 03:17:41 +0300135 }
136
Damien George999cedb2015-11-27 17:01:44 +0000137 t->items[0] = MP_OBJ_FROM_PTR(mp_obj_get_type(cur_exc));
Paul Sokolovsky8b85d142015-04-25 03:17:41 +0300138 t->items[1] = cur_exc;
139 t->items[2] = mp_const_none;
Damien George999cedb2015-11-27 17:01:44 +0000140 return MP_OBJ_FROM_PTR(t);
Paul Sokolovsky8b85d142015-04-25 03:17:41 +0300141}
142MP_DEFINE_CONST_FUN_OBJ_0(mp_sys_exc_info_obj, mp_sys_exc_info);
143#endif
144
Damien Georgecbf76742015-11-27 13:38:15 +0000145STATIC const mp_rom_map_elem_t mp_module_sys_globals_table[] = {
146 { MP_ROM_QSTR(MP_QSTR___name__), MP_ROM_QSTR(MP_QSTR_sys) },
Paul Sokolovskyd99e9082014-05-10 16:50:45 +0300147
Damien Georgecbf76742015-11-27 13:38:15 +0000148 { MP_ROM_QSTR(MP_QSTR_path), MP_ROM_PTR(&MP_STATE_VM(mp_sys_path_obj)) },
149 { MP_ROM_QSTR(MP_QSTR_argv), MP_ROM_PTR(&MP_STATE_VM(mp_sys_argv_obj)) },
150 { MP_ROM_QSTR(MP_QSTR_version), MP_ROM_PTR(&version_obj) },
151 { MP_ROM_QSTR(MP_QSTR_version_info), MP_ROM_PTR(&mp_sys_version_info_obj) },
152 { MP_ROM_QSTR(MP_QSTR_implementation), MP_ROM_PTR(&mp_sys_implementation_obj) },
Paul Sokolovsky72bd1722015-11-21 15:32:00 +0200153 #ifdef MICROPY_PY_SYS_PLATFORM
Damien Georgecbf76742015-11-27 13:38:15 +0000154 { MP_ROM_QSTR(MP_QSTR_platform), MP_ROM_PTR(&platform_obj) },
Paul Sokolovsky72bd1722015-11-21 15:32:00 +0200155 #endif
Damien George30dd23a2014-08-10 17:50:28 +0100156 /// \constant byteorder - the byte order of the system ("little" or "big")
Paul Sokolovsky72bd1722015-11-21 15:32:00 +0200157 #if MP_ENDIANNESS_LITTLE
Damien Georgecbf76742015-11-27 13:38:15 +0000158 { MP_ROM_QSTR(MP_QSTR_byteorder), MP_ROM_QSTR(MP_QSTR_little) },
Paul Sokolovsky72bd1722015-11-21 15:32:00 +0200159 #else
Damien Georgecbf76742015-11-27 13:38:15 +0000160 { MP_ROM_QSTR(MP_QSTR_byteorder), MP_ROM_QSTR(MP_QSTR_big) },
Paul Sokolovsky72bd1722015-11-21 15:32:00 +0200161 #endif
162
163 #if MICROPY_PY_SYS_MAXSIZE
Paul Sokolovsky4e0eeeb2014-07-03 16:50:11 +0300164 #if MICROPY_LONGINT_IMPL == MICROPY_LONGINT_IMPL_NONE
165 // INT_MAX is not representable as small int, as we know that small int
166 // takes one bit for tag. So, we have little choice but to provide this
167 // value. Apps also should be careful to not try to compare sys.maxsize
168 // with some number (which may not fit in available int size), but instead
169 // count number of significant bits in sys.maxsize.
Damien Georgecbf76742015-11-27 13:38:15 +0000170 { MP_ROM_QSTR(MP_QSTR_maxsize), MP_OBJ_NEW_SMALL_INT(INT_MAX >> 1) },
Paul Sokolovsky4e0eeeb2014-07-03 16:50:11 +0300171 #else
Damien Georgecbf76742015-11-27 13:38:15 +0000172 { MP_ROM_QSTR(MP_QSTR_maxsize), MP_ROM_PTR(&mp_maxsize_obj) },
Paul Sokolovsky4e0eeeb2014-07-03 16:50:11 +0300173 #endif
Paul Sokolovsky72bd1722015-11-21 15:32:00 +0200174 #endif
Paul Sokolovsky4e0eeeb2014-07-03 16:50:11 +0300175
Paul Sokolovsky72bd1722015-11-21 15:32:00 +0200176 #if MICROPY_PY_SYS_EXIT
Damien George30dd23a2014-08-10 17:50:28 +0100177 // documented per-port
Damien Georgecbf76742015-11-27 13:38:15 +0000178 { MP_ROM_QSTR(MP_QSTR_exit), MP_ROM_PTR(&mp_sys_exit_obj) },
Paul Sokolovsky72bd1722015-11-21 15:32:00 +0200179 #endif
Damien George89755ae2014-05-11 17:35:43 +0100180
Paul Sokolovsky72bd1722015-11-21 15:32:00 +0200181 #if MICROPY_PY_SYS_STDFILES
Damien George30dd23a2014-08-10 17:50:28 +0100182 // documented per-port
Damien Georgecbf76742015-11-27 13:38:15 +0000183 { MP_ROM_QSTR(MP_QSTR_stdin), MP_ROM_PTR(&mp_sys_stdin_obj) },
184 { MP_ROM_QSTR(MP_QSTR_stdout), MP_ROM_PTR(&mp_sys_stdout_obj) },
185 { MP_ROM_QSTR(MP_QSTR_stderr), MP_ROM_PTR(&mp_sys_stderr_obj) },
Paul Sokolovsky72bd1722015-11-21 15:32:00 +0200186 #endif
Paul Sokolovsky46c3ab22014-12-06 14:29:09 +0200187
Paul Sokolovsky1a1d11f2015-12-05 00:09:10 +0200188 #if MICROPY_PY_SYS_MODULES
189 { MP_OBJ_NEW_QSTR(MP_QSTR_modules), MP_OBJ_FROM_PTR(&MP_STATE_VM(mp_loaded_modules_dict)) },
190 #endif
Paul Sokolovsky8b85d142015-04-25 03:17:41 +0300191 #if MICROPY_PY_SYS_EXC_INFO
Damien Georgecbf76742015-11-27 13:38:15 +0000192 { MP_ROM_QSTR(MP_QSTR_exc_info), MP_ROM_PTR(&mp_sys_exc_info_obj) },
Paul Sokolovsky8b85d142015-04-25 03:17:41 +0300193 #endif
194
Paul Sokolovsky46c3ab22014-12-06 14:29:09 +0200195 /*
196 * Extensions to CPython
197 */
198
Damien Georgecbf76742015-11-27 13:38:15 +0000199 { MP_ROM_QSTR(MP_QSTR_print_exception), MP_ROM_PTR(&mp_sys_print_exception_obj) },
Paul Sokolovsky5500cde2014-04-13 06:43:18 +0300200};
201
Damien George3b603f22014-11-29 14:39:27 +0000202STATIC MP_DEFINE_CONST_DICT(mp_module_sys_globals, mp_module_sys_globals_table);
Paul Sokolovsky5500cde2014-04-13 06:43:18 +0300203
204const mp_obj_module_t mp_module_sys = {
205 .base = { &mp_type_module },
206 .name = MP_QSTR_sys,
207 .globals = (mp_obj_dict_t*)&mp_module_sys_globals,
208};
209
210#endif