mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-25 02:30:57 +00:00 
			
		
		
		
	- Python static hello world now 1.8mb
- Python static fully loaded now 10mb
- Python HTTPS client now uses MbedTLS
- Python REPL now completes import stmts
- Increase stack size for Python for now
- Begin synthesizing posixpath and ntpath
- Restore Python \N{UNICODE NAME} support
- Restore Python NFKD symbol normalization
- Add optimized code path for Intel SHA-NI
- Get more Python unit tests passing faster
- Get Python help() pagination working on NT
- Python hashlib now supports MbedTLS PBKDF2
- Make memcpy/memmove/memcmp/bcmp/etc. faster
- Add Mersenne Twister and Vigna to LIBC_RAND
- Provide privileged __printf() for error code
- Fix zipos opendir() so that it reports ENOTDIR
- Add basic chmod() implementation for Windows NT
- Add Cosmo's best functions to Python cosmo module
- Pin function trace indent depth to that of caller
- Show memory diagram on invalid access in MODE=dbg
- Differentiate stack overflow on crash in MODE=dbg
- Add stb_truetype and tools for analyzing font files
- Upgrade to UNICODE 13 and reduce its binary footprint
- COMPILE.COM now logs resource usage of build commands
- Start implementing basic poll() support on bare metal
- Set getauxval(AT_EXECFN) to GetModuleFileName() on NT
- Add descriptions to strerror() in non-TINY build modes
- Add COUNTBRANCH() macro to help with micro-optimizations
- Make error / backtrace / asan / memory code more unbreakable
- Add fast perfect C implementation of μ-Law and a-Law audio codecs
- Make strtol() functions consistent with other libc implementations
- Improve Linenoise implementation (see also github.com/jart/bestline)
- COMPILE.COM now suppresses stdout/stderr of successful build commands
		
	
			
		
			
				
	
	
		
			154 lines
		
	
	
	
		
			6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			154 lines
		
	
	
	
		
			6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:4;tab-width:8;coding:utf-8 -*-│
 | |
| │vi: set net ft=c ts=4 sts=4 sw=4 fenc=utf-8                                :vi│
 | |
| ╞══════════════════════════════════════════════════════════════════════════════╡
 | |
| │ Python 3                                                                     │
 | |
| │ https://docs.python.org/3/license.html                                       │
 | |
| ╚─────────────────────────────────────────────────────────────────────────────*/
 | |
| #include "third_party/python/Include/Python-ast.h"
 | |
| #include "third_party/python/Include/code.h"
 | |
| #include "third_party/python/Include/graminit.h"
 | |
| #include "third_party/python/Include/node.h"
 | |
| #include "third_party/python/Include/objimpl.h"
 | |
| #include "third_party/python/Include/pyerrors.h"
 | |
| #include "third_party/python/Include/symtable.h"
 | |
| #include "third_party/python/Include/token.h"
 | |
| #include "third_party/python/Include/unicodeobject.h"
 | |
| /* clang-format off */
 | |
| 
 | |
| #define UNDEFINED_FUTURE_FEATURE "future feature %.100s is not defined"
 | |
| #define ERR_LATE_FUTURE \
 | |
| "from __future__ imports must occur at the beginning of the file"
 | |
| 
 | |
| static int
 | |
| future_check_features(PyFutureFeatures *ff, stmt_ty s, PyObject *filename)
 | |
| {
 | |
|     int i;
 | |
|     asdl_seq *names;
 | |
|     assert(s->kind == ImportFrom_kind);
 | |
|     names = s->v.ImportFrom.names;
 | |
|     for (i = 0; i < asdl_seq_LEN(names); i++) {
 | |
|         alias_ty name = (alias_ty)asdl_seq_GET(names, i);
 | |
|         const char *feature = PyUnicode_AsUTF8(name->name);
 | |
|         if (!feature)
 | |
|             return 0;
 | |
|         if (strcmp(feature, FUTURE_NESTED_SCOPES) == 0) {
 | |
|             continue;
 | |
|         } else if (strcmp(feature, FUTURE_GENERATORS) == 0) {
 | |
|             continue;
 | |
|         } else if (strcmp(feature, FUTURE_DIVISION) == 0) {
 | |
|             continue;
 | |
|         } else if (strcmp(feature, FUTURE_ABSOLUTE_IMPORT) == 0) {
 | |
|             continue;
 | |
|         } else if (strcmp(feature, FUTURE_WITH_STATEMENT) == 0) {
 | |
|             continue;
 | |
|         } else if (strcmp(feature, FUTURE_PRINT_FUNCTION) == 0) {
 | |
|             continue;
 | |
|         } else if (strcmp(feature, FUTURE_UNICODE_LITERALS) == 0) {
 | |
|             continue;
 | |
|         } else if (strcmp(feature, FUTURE_BARRY_AS_BDFL) == 0) {
 | |
|             ff->ff_features |= CO_FUTURE_BARRY_AS_BDFL;
 | |
|         } else if (strcmp(feature, FUTURE_GENERATOR_STOP) == 0) {
 | |
|             ff->ff_features |= CO_FUTURE_GENERATOR_STOP;
 | |
|         } else if (strcmp(feature, "braces") == 0) {
 | |
|             PyErr_SetString(PyExc_SyntaxError,
 | |
|                             "not a chance");
 | |
|             PyErr_SyntaxLocationObject(filename, s->lineno, s->col_offset);
 | |
|             return 0;
 | |
|         } else {
 | |
|             PyErr_Format(PyExc_SyntaxError,
 | |
|                          UNDEFINED_FUTURE_FEATURE, feature);
 | |
|             PyErr_SyntaxLocationObject(filename, s->lineno, s->col_offset);
 | |
|             return 0;
 | |
|         }
 | |
|     }
 | |
|     return 1;
 | |
| }
 | |
| 
 | |
| static int
 | |
| future_parse(PyFutureFeatures *ff, mod_ty mod, PyObject *filename)
 | |
| {
 | |
|     int i, done = 0, prev_line = 0;
 | |
|     stmt_ty first;
 | |
|     if (!(mod->kind == Module_kind || mod->kind == Interactive_kind))
 | |
|         return 1;
 | |
|     if (asdl_seq_LEN(mod->v.Module.body) == 0)
 | |
|         return 1;
 | |
|     /* A subsequent pass will detect future imports that don't
 | |
|        appear at the beginning of the file.  There's one case,
 | |
|        however, that is easier to handle here: A series of imports
 | |
|        joined by semi-colons, where the first import is a future
 | |
|        statement but some subsequent import has the future form
 | |
|        but is preceded by a regular import.
 | |
|     */
 | |
|     i = 0;
 | |
|     first = (stmt_ty)asdl_seq_GET(mod->v.Module.body, i);
 | |
|     if (first->kind == Expr_kind
 | |
|         && (first->v.Expr.value->kind == Str_kind
 | |
|             || (first->v.Expr.value->kind == Constant_kind
 | |
|                 && PyUnicode_CheckExact(first->v.Expr.value->v.Constant.value))))
 | |
|         i++;
 | |
|     for (; i < asdl_seq_LEN(mod->v.Module.body); i++) {
 | |
|         stmt_ty s = (stmt_ty)asdl_seq_GET(mod->v.Module.body, i);
 | |
|         if (done && s->lineno > prev_line)
 | |
|             return 1;
 | |
|         prev_line = s->lineno;
 | |
|         /* The tests below will return from this function unless it is
 | |
|            still possible to find a future statement.  The only things
 | |
|            that can precede a future statement are another future
 | |
|            statement and a doc string.
 | |
|         */
 | |
|         if (s->kind == ImportFrom_kind) {
 | |
|             identifier modname = s->v.ImportFrom.module;
 | |
|             if (modname &&
 | |
|                 _PyUnicode_EqualToASCIIString(modname, "__future__")) {
 | |
|                 if (done) {
 | |
|                     PyErr_SetString(PyExc_SyntaxError,
 | |
|                                     ERR_LATE_FUTURE);
 | |
|                     PyErr_SyntaxLocationObject(filename, s->lineno, s->col_offset);
 | |
|                     return 0;
 | |
|                 }
 | |
|                 if (!future_check_features(ff, s, filename))
 | |
|                     return 0;
 | |
|                 ff->ff_lineno = s->lineno;
 | |
|             }
 | |
|             else {
 | |
|                 done = 1;
 | |
|             }
 | |
|         }
 | |
|         else {
 | |
|             done = 1;
 | |
|         }
 | |
|     }
 | |
|     return 1;
 | |
| }
 | |
| 
 | |
| PyFutureFeatures *
 | |
| PyFuture_FromASTObject(mod_ty mod, PyObject *filename)
 | |
| {
 | |
|     PyFutureFeatures *ff;
 | |
|     ff = (PyFutureFeatures *)PyObject_Malloc(sizeof(PyFutureFeatures));
 | |
|     if (ff == NULL) {
 | |
|         PyErr_NoMemory();
 | |
|         return NULL;
 | |
|     }
 | |
|     ff->ff_features = 0;
 | |
|     ff->ff_lineno = -1;
 | |
|     if (!future_parse(ff, mod, filename)) {
 | |
|         PyObject_Free(ff);
 | |
|         return NULL;
 | |
|     }
 | |
|     return ff;
 | |
| }
 | |
| 
 | |
| PyFutureFeatures *
 | |
| PyFuture_FromAST(mod_ty mod, const char *filename_str)
 | |
| {
 | |
|     PyFutureFeatures *ff;
 | |
|     PyObject *filename;
 | |
|     filename = PyUnicode_DecodeFSDefault(filename_str);
 | |
|     if (filename == NULL)
 | |
|         return NULL;
 | |
|     ff = PyFuture_FromASTObject(mod, filename);
 | |
|     Py_DECREF(filename);
 | |
|     return ff;
 | |
| }
 |