mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-26 11:10:58 +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
		
	
			
		
			
				
	
	
		
			284 lines
		
	
	
	
		
			8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			284 lines
		
	
	
	
		
			8 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/musl/passwd.h"
 | |
| #include "third_party/python/Include/abstract.h"
 | |
| #include "third_party/python/Include/bytesobject.h"
 | |
| #include "third_party/python/Include/dictobject.h"
 | |
| #include "third_party/python/Include/import.h"
 | |
| #include "third_party/python/Include/listobject.h"
 | |
| #include "third_party/python/Include/modsupport.h"
 | |
| #include "third_party/python/Include/object.h"
 | |
| #include "third_party/python/Include/pyerrors.h"
 | |
| #include "third_party/python/Include/pymacro.h"
 | |
| #include "third_party/python/Include/structseq.h"
 | |
| #include "third_party/python/Include/unicodeobject.h"
 | |
| #include "third_party/python/Include/warnings.h"
 | |
| #include "third_party/python/Include/yoink.h"
 | |
| #include "third_party/python/Modules/clinic/grpmodule.inc"
 | |
| #include "third_party/python/Modules/posixmodule.h"
 | |
| /* clang-format off */
 | |
| 
 | |
| PYTHON_PROVIDE("grp");
 | |
| PYTHON_PROVIDE("grp.getgrall");
 | |
| PYTHON_PROVIDE("grp.getgrgid");
 | |
| PYTHON_PROVIDE("grp.getgrnam");
 | |
| PYTHON_PROVIDE("grp.struct_group");
 | |
| 
 | |
| /* UNIX group file access module */
 | |
| 
 | |
| /*[clinic input]
 | |
| module grp
 | |
| [clinic start generated code]*/
 | |
| /*[clinic end generated code: output=da39a3ee5e6b4b0d input=cade63f2ed1bd9f8]*/
 | |
| 
 | |
| static PyStructSequence_Field struct_group_type_fields[] = {
 | |
|     {"gr_name", PyDoc_STR("group name")},
 | |
|     {"gr_passwd", PyDoc_STR("password")},
 | |
|     {"gr_gid", PyDoc_STR("group id")},
 | |
|     {"gr_mem", PyDoc_STR("group members")},
 | |
|     {0}
 | |
| };
 | |
| 
 | |
| PyDoc_STRVAR(struct_group__doc__,
 | |
| "grp.struct_group: Results from getgr*() routines.\n\n\
 | |
| This object may be accessed either as a tuple of\n\
 | |
|   (gr_name,gr_passwd,gr_gid,gr_mem)\n\
 | |
| or via the object attributes as named in the above tuple.\n");
 | |
| 
 | |
| static PyStructSequence_Desc struct_group_type_desc = {
 | |
|    "grp.struct_group",
 | |
|    struct_group__doc__,
 | |
|    struct_group_type_fields,
 | |
|    4,
 | |
| };
 | |
| 
 | |
| 
 | |
| static int initialized;
 | |
| static PyTypeObject StructGrpType;
 | |
| 
 | |
| static PyObject *
 | |
| mkgrent(struct group *p)
 | |
| {
 | |
|     int setIndex = 0;
 | |
|     PyObject *v = PyStructSequence_New(&StructGrpType), *w;
 | |
|     char **member;
 | |
| 
 | |
|     if (v == NULL)
 | |
|         return NULL;
 | |
| 
 | |
|     if ((w = PyList_New(0)) == NULL) {
 | |
|         Py_DECREF(v);
 | |
|         return NULL;
 | |
|     }
 | |
|     for (member = p->gr_mem; *member != NULL; member++) {
 | |
|         PyObject *x = PyUnicode_DecodeFSDefault(*member);
 | |
|         if (x == NULL || PyList_Append(w, x) != 0) {
 | |
|             Py_XDECREF(x);
 | |
|             Py_DECREF(w);
 | |
|             Py_DECREF(v);
 | |
|             return NULL;
 | |
|         }
 | |
|         Py_DECREF(x);
 | |
|     }
 | |
| 
 | |
| #define SET(i,val) PyStructSequence_SET_ITEM(v, i, val)
 | |
|     SET(setIndex++, PyUnicode_DecodeFSDefault(p->gr_name));
 | |
|     if (p->gr_passwd)
 | |
|             SET(setIndex++, PyUnicode_DecodeFSDefault(p->gr_passwd));
 | |
|     else {
 | |
|             SET(setIndex++, Py_None);
 | |
|             Py_INCREF(Py_None);
 | |
|     }
 | |
|     SET(setIndex++, _PyLong_FromGid(p->gr_gid));
 | |
|     SET(setIndex++, w);
 | |
| #undef SET
 | |
| 
 | |
|     if (PyErr_Occurred()) {
 | |
|         Py_DECREF(v);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     return v;
 | |
| }
 | |
| 
 | |
| /*[clinic input]
 | |
| grp.getgrgid
 | |
| 
 | |
|     id: object
 | |
| 
 | |
| Return the group database entry for the given numeric group ID.
 | |
| 
 | |
| If id is not valid, raise KeyError.
 | |
| [clinic start generated code]*/
 | |
| 
 | |
| static PyObject *
 | |
| grp_getgrgid_impl(PyObject *module, PyObject *id)
 | |
| /*[clinic end generated code: output=30797c289504a1ba input=15fa0e2ccf5cda25]*/
 | |
| {
 | |
|     PyObject *py_int_id;
 | |
|     gid_t gid;
 | |
|     struct group *p;
 | |
| 
 | |
|     if (!_Py_Gid_Converter(id, &gid)) {
 | |
|         if (!PyErr_ExceptionMatches(PyExc_TypeError)) {
 | |
|             return NULL;
 | |
|         }
 | |
|         PyErr_Clear();
 | |
|         if (PyErr_WarnFormat(PyExc_DeprecationWarning, 1,
 | |
|                              "group id must be int, not %.200",
 | |
|                              id->ob_type->tp_name) < 0) {
 | |
|             return NULL;
 | |
|         }
 | |
|         py_int_id = PyNumber_Long(id);
 | |
|         if (!py_int_id)
 | |
|             return NULL;
 | |
|         if (!_Py_Gid_Converter(py_int_id, &gid)) {
 | |
|             Py_DECREF(py_int_id);
 | |
|             return NULL;
 | |
|         }
 | |
|         Py_DECREF(py_int_id);
 | |
|     }
 | |
| 
 | |
|     if ((p = getgrgid(gid)) == NULL) {
 | |
|         PyObject *gid_obj = _PyLong_FromGid(gid);
 | |
|         if (gid_obj == NULL)
 | |
|             return NULL;
 | |
|         PyErr_Format(PyExc_KeyError, "getgrgid(): gid not found: %S", gid_obj);
 | |
|         Py_DECREF(gid_obj);
 | |
|         return NULL;
 | |
|     }
 | |
|     return mkgrent(p);
 | |
| }
 | |
| 
 | |
| /*[clinic input]
 | |
| grp.getgrnam
 | |
| 
 | |
|     name: unicode
 | |
| 
 | |
| Return the group database entry for the given group name.
 | |
| 
 | |
| If name is not valid, raise KeyError.
 | |
| [clinic start generated code]*/
 | |
| 
 | |
| static PyObject *
 | |
| grp_getgrnam_impl(PyObject *module, PyObject *name)
 | |
| /*[clinic end generated code: output=67905086f403c21c input=08ded29affa3c863]*/
 | |
| {
 | |
|     char *name_chars;
 | |
|     struct group *p;
 | |
|     PyObject *bytes, *retval = NULL;
 | |
| 
 | |
|     if ((bytes = PyUnicode_EncodeFSDefault(name)) == NULL)
 | |
|         return NULL;
 | |
|     /* check for embedded null bytes */
 | |
|     if (PyBytes_AsStringAndSize(bytes, &name_chars, NULL) == -1)
 | |
|         goto out;
 | |
| 
 | |
|     if ((p = getgrnam(name_chars)) == NULL) {
 | |
|         PyErr_Format(PyExc_KeyError, "getgrnam(): name not found: %R", name);
 | |
|         goto out;
 | |
|     }
 | |
|     retval = mkgrent(p);
 | |
| out:
 | |
|     Py_DECREF(bytes);
 | |
|     return retval;
 | |
| }
 | |
| 
 | |
| /*[clinic input]
 | |
| grp.getgrall
 | |
| 
 | |
| Return a list of all available group entries, in arbitrary order.
 | |
| 
 | |
| An entry whose name starts with '+' or '-' represents an instruction
 | |
| to use YP/NIS and may not be accessible via getgrnam or getgrgid.
 | |
| [clinic start generated code]*/
 | |
| 
 | |
| static PyObject *
 | |
| grp_getgrall_impl(PyObject *module)
 | |
| /*[clinic end generated code: output=585dad35e2e763d7 input=d7df76c825c367df]*/
 | |
| {
 | |
|     PyObject *d;
 | |
|     struct group *p;
 | |
| 
 | |
|     if ((d = PyList_New(0)) == NULL)
 | |
|         return NULL;
 | |
|     setgrent();
 | |
|     while ((p = getgrent()) != NULL) {
 | |
|         PyObject *v = mkgrent(p);
 | |
|         if (v == NULL || PyList_Append(d, v) != 0) {
 | |
|             Py_XDECREF(v);
 | |
|             Py_DECREF(d);
 | |
|             endgrent();
 | |
|             return NULL;
 | |
|         }
 | |
|         Py_DECREF(v);
 | |
|     }
 | |
|     endgrent();
 | |
|     return d;
 | |
| }
 | |
| 
 | |
| static PyMethodDef grp_methods[] = {
 | |
|     GRP_GETGRGID_METHODDEF
 | |
|     GRP_GETGRNAM_METHODDEF
 | |
|     GRP_GETGRALL_METHODDEF
 | |
|     {NULL, NULL}
 | |
| };
 | |
| 
 | |
| PyDoc_STRVAR(grp__doc__,
 | |
| "Access to the Unix group database.\n\
 | |
| \n\
 | |
| Group entries are reported as 4-tuples containing the following fields\n\
 | |
| from the group database, in order:\n\
 | |
| \n\
 | |
|   gr_name   - name of the group\n\
 | |
|   gr_passwd - group password (encrypted); often empty\n\
 | |
|   gr_gid    - numeric ID of the group\n\
 | |
|   gr_mem    - list of members\n\
 | |
| \n\
 | |
| The gid is an integer, name and password are strings.  (Note that most\n\
 | |
| users are not explicitly listed as members of the groups they are in\n\
 | |
| according to the password database.  Check both databases to get\n\
 | |
| complete membership information.)");
 | |
| 
 | |
| 
 | |
| 
 | |
| static struct PyModuleDef grpmodule = {
 | |
|         PyModuleDef_HEAD_INIT,
 | |
|         "grp",
 | |
|         grp__doc__,
 | |
|         -1,
 | |
|         grp_methods,
 | |
|         NULL,
 | |
|         NULL,
 | |
|         NULL,
 | |
|         NULL
 | |
| };
 | |
| 
 | |
| PyMODINIT_FUNC
 | |
| PyInit_grp(void)
 | |
| {
 | |
|     PyObject *m, *d;
 | |
|     m = PyModule_Create(&grpmodule);
 | |
|     if (m == NULL)
 | |
|         return NULL;
 | |
|     d = PyModule_GetDict(m);
 | |
|     if (!initialized) {
 | |
|         if (PyStructSequence_InitType2(&StructGrpType,
 | |
|                                        &struct_group_type_desc) < 0)
 | |
|             return NULL;
 | |
|     }
 | |
|     if (PyDict_SetItemString(d, "struct_group",
 | |
|                              (PyObject *)&StructGrpType) < 0)
 | |
|         return NULL;
 | |
|     initialized = 1;
 | |
|     return m;
 | |
| }
 | |
| 
 | |
| _Section(".rodata.pytab.1") const struct _inittab _PyImport_Inittab_grp = {
 | |
|     "grp",
 | |
|     PyInit_grp,
 | |
| };
 |