Commit 6dcb5422 authored by Victor Stinner's avatar Victor Stinner Committed by GitHub

bpo-36142: Add _PyPreConfig_ReadFromArgv() (GH-12173)

The new function is now responsible to parse -E and -I command line
arguments.
parent cad1f747
...@@ -44,12 +44,13 @@ PyAPI_FUNC(void) _PyPreConfig_SetGlobalConfig(const _PyPreConfig *config); ...@@ -44,12 +44,13 @@ PyAPI_FUNC(void) _PyPreConfig_SetGlobalConfig(const _PyPreConfig *config);
PyAPI_FUNC(_PyInitError) _PyPreConfig_Read(_PyPreConfig *config); PyAPI_FUNC(_PyInitError) _PyPreConfig_Read(_PyPreConfig *config);
PyAPI_FUNC(int) _PyPreConfig_AsDict(const _PyPreConfig *config, PyAPI_FUNC(int) _PyPreConfig_AsDict(const _PyPreConfig *config,
PyObject *dict); PyObject *dict);
PyAPI_FUNC(_PyInitError) _PyPreConfig_ReadFromArgv(_PyPreConfig *config,
const _PyArgv *args);
PyAPI_FUNC(void) _PyPreConfig_Write(const _PyPreConfig *config);
/* --- _PyCoreConfig ---------------------------------------------- */ /* --- _PyCoreConfig ---------------------------------------------- */
PyAPI_FUNC(_PyInitError) _PyCoreConfig_Read(_PyCoreConfig *config);
PyAPI_FUNC(void) _PyCoreConfig_Clear(_PyCoreConfig *); PyAPI_FUNC(void) _PyCoreConfig_Clear(_PyCoreConfig *);
PyAPI_FUNC(int) _PyCoreConfig_Copy( PyAPI_FUNC(int) _PyCoreConfig_Copy(
_PyCoreConfig *config, _PyCoreConfig *config,
...@@ -67,8 +68,11 @@ PyAPI_FUNC(int) _PyCoreConfig_GetEnvDup( ...@@ -67,8 +68,11 @@ PyAPI_FUNC(int) _PyCoreConfig_GetEnvDup(
wchar_t **dest, wchar_t **dest,
wchar_t *wname, wchar_t *wname,
char *name); char *name);
PyAPI_FUNC(_PyInitError) _PyCoreConfig_Read(_PyCoreConfig *config,
const _PyPreConfig *preconfig);
PyAPI_FUNC(_PyInitError) _PyCoreConfig_ReadFromArgv(_PyCoreConfig *config, PyAPI_FUNC(_PyInitError) _PyCoreConfig_ReadFromArgv(_PyCoreConfig *config,
const _PyArgv *args); const _PyArgv *args,
const _PyPreConfig *preconfig);
PyAPI_FUNC(void) _PyCoreConfig_Write(const _PyCoreConfig *config); PyAPI_FUNC(void) _PyCoreConfig_Write(const _PyCoreConfig *config);
#ifdef __cplusplus #ifdef __cplusplus
......
...@@ -17,7 +17,6 @@ typedef struct { ...@@ -17,7 +17,6 @@ typedef struct {
int val; int val;
} _PyOS_LongOption; } _PyOS_LongOption;
extern int _PyOS_GetOpt(int argc, wchar_t **argv, wchar_t *optstring, extern int _PyOS_GetOpt(int argc, wchar_t **argv, int *longindex);
const _PyOS_LongOption *longopts, int *longindex);
#endif /* !Py_INTERNAL_PYGETOPT_H */ #endif /* !Py_INTERNAL_PYGETOPT_H */
...@@ -286,20 +286,32 @@ _PyMainInterpreterConfig_Read(_PyMainInterpreterConfig *main_config, ...@@ -286,20 +286,32 @@ _PyMainInterpreterConfig_Read(_PyMainInterpreterConfig *main_config,
/* --- pymain_init() ---------------------------------------------- */ /* --- pymain_init() ---------------------------------------------- */
static void static _PyInitError
config_clear(_PyCoreConfig *config) preconfig_read_write(_PyPreConfig *config, const _PyArgv *args)
{ {
_PyInitError err;
PyMemAllocatorEx old_alloc; PyMemAllocatorEx old_alloc;
_PyMem_SetDefaultAllocator(PYMEM_DOMAIN_RAW, &old_alloc); _PyMem_SetDefaultAllocator(PYMEM_DOMAIN_RAW, &old_alloc);
_PyCoreConfig_Clear(config); _PyPreConfig_GetGlobalConfig(config);
err = _PyPreConfig_ReadFromArgv(config, args);
PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &old_alloc); PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &old_alloc);
if (_Py_INIT_FAILED(err)) {
return err;
}
_PyPreConfig_Write(config);
return _Py_INIT_OK();
} }
static _PyInitError static _PyInitError
config_read_write(_PyCoreConfig *config, const _PyArgv *args) config_read_write(_PyCoreConfig *config, const _PyArgv *args,
const _PyPreConfig *preconfig)
{ {
_PyInitError err; _PyInitError err;
...@@ -308,7 +320,7 @@ config_read_write(_PyCoreConfig *config, const _PyArgv *args) ...@@ -308,7 +320,7 @@ config_read_write(_PyCoreConfig *config, const _PyArgv *args)
_PyCoreConfig_GetGlobalConfig(config); _PyCoreConfig_GetGlobalConfig(config);
err = _PyCoreConfig_ReadFromArgv(config, args); err = _PyCoreConfig_ReadFromArgv(config, args, preconfig);
PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &old_alloc); PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &old_alloc);
...@@ -344,6 +356,7 @@ static _PyInitError ...@@ -344,6 +356,7 @@ static _PyInitError
pymain_init(const _PyArgv *args, PyInterpreterState **interp_p) pymain_init(const _PyArgv *args, PyInterpreterState **interp_p)
{ {
_PyInitError err; _PyInitError err;
PyMemAllocatorEx old_alloc;
err = _PyRuntime_Initialize(); err = _PyRuntime_Initialize();
if (_Py_INIT_FAILED(err)) { if (_Py_INIT_FAILED(err)) {
...@@ -359,10 +372,18 @@ pymain_init(const _PyArgv *args, PyInterpreterState **interp_p) ...@@ -359,10 +372,18 @@ pymain_init(const _PyArgv *args, PyInterpreterState **interp_p)
fedisableexcept(FE_OVERFLOW); fedisableexcept(FE_OVERFLOW);
#endif #endif
_PyPreConfig local_preconfig = _PyPreConfig_INIT;
_PyPreConfig *preconfig = &local_preconfig;
_PyCoreConfig local_config = _PyCoreConfig_INIT; _PyCoreConfig local_config = _PyCoreConfig_INIT;
_PyCoreConfig *config = &local_config; _PyCoreConfig *config = &local_config;
err = config_read_write(config, args); err = preconfig_read_write(preconfig, args);
if (_Py_INIT_FAILED(err)) {
goto done;
}
err = config_read_write(config, args, preconfig);
if (_Py_INIT_FAILED(err)) { if (_Py_INIT_FAILED(err)) {
goto done; goto done;
} }
...@@ -382,7 +403,12 @@ pymain_init(const _PyArgv *args, PyInterpreterState **interp_p) ...@@ -382,7 +403,12 @@ pymain_init(const _PyArgv *args, PyInterpreterState **interp_p)
err = _Py_INIT_OK(); err = _Py_INIT_OK();
done: done:
config_clear(config); _PyMem_SetDefaultAllocator(PYMEM_DOMAIN_RAW, &old_alloc);
_PyPreConfig_Clear(preconfig);
_PyCoreConfig_Clear(config);
PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &old_alloc);
return err; return err;
} }
......
...@@ -23,13 +23,6 @@ ...@@ -23,13 +23,6 @@
/* --- Command line options --------------------------------------- */ /* --- Command line options --------------------------------------- */
#define PROGRAM_OPTS L"bBc:dEhiIJm:OqRsStuvVW:xX:?"
static const _PyOS_LongOption longoptions[] = {
{L"check-hash-based-pycs", 1, 0},
{NULL, 0, 0},
};
/* Short usage message (with %s for argv0) */ /* Short usage message (with %s for argv0) */
static const char usage_line[] = static const char usage_line[] =
"usage: %ls [option] ... [-c cmd | -m mod | file | -] [arg] ...\n"; "usage: %ls [option] ... [-c cmd | -m mod | file | -] [arg] ...\n";
...@@ -1483,29 +1476,62 @@ config_init_fs_encoding(_PyCoreConfig *config) ...@@ -1483,29 +1476,62 @@ config_init_fs_encoding(_PyCoreConfig *config)
} }
/* Read configuration settings from standard locations static _PyInitError
* _PyCoreConfig_ReadPreConfig(_PyCoreConfig *config)
* This function doesn't make any changes to the interpreter state - it {
* merely populates any missing configuration settings. This allows an _PyInitError err;
* embedding application to completely override a config option by _PyPreConfig local_preconfig = _PyPreConfig_INIT;
* setting it before calling this function, or else modify the default _PyPreConfig_GetGlobalConfig(&local_preconfig);
* setting before passing the fully populated config to Py_EndInitialization.
* if (_PyPreConfig_Copy(&local_preconfig, &config->preconfig) < 0) {
* More advanced selective initialization tricks are possible by calling err = _Py_INIT_NO_MEMORY();
* this function multiple times with various preconfigured settings. goto done;
*/ }
err = _PyPreConfig_Read(&local_preconfig);
if (_Py_INIT_FAILED(err)) {
goto done;
}
if (_PyPreConfig_Copy(&config->preconfig, &local_preconfig) < 0) {
err = _Py_INIT_NO_MEMORY();
goto done;
}
err = _Py_INIT_OK();
done:
_PyPreConfig_Clear(&local_preconfig);
return err;
}
/* Read the configuration into _PyCoreConfig and initialize the LC_CTYPE
locale: enable UTF-8 mode (PEP 540) and/or coerce the C locale (PEP 538).
Read the configuration from:
* Environment variables
* Py_xxx global configuration variables
See _PyCoreConfig_ReadFromArgv() to parse also command line arguments. */
_PyInitError _PyInitError
_PyCoreConfig_Read(_PyCoreConfig *config) _PyCoreConfig_Read(_PyCoreConfig *config, const _PyPreConfig *preconfig)
{ {
_PyInitError err; _PyInitError err;
_PyCoreConfig_GetGlobalConfig(config); _PyCoreConfig_GetGlobalConfig(config);
err = _PyPreConfig_Read(&config->preconfig); if (preconfig != NULL) {
if (_PyPreConfig_Copy(&config->preconfig, preconfig) < 0) {
return _Py_INIT_NO_MEMORY();
}
}
else {
err = _PyCoreConfig_ReadPreConfig(config);
if (_Py_INIT_FAILED(err)) { if (_Py_INIT_FAILED(err)) {
return err; return err;
} }
}
assert(config->preconfig.use_environment >= 0); assert(config->preconfig.use_environment >= 0);
...@@ -1851,8 +1877,7 @@ config_parse_cmdline(_PyCoreConfig *config, _PyCmdline *cmdline, ...@@ -1851,8 +1877,7 @@ config_parse_cmdline(_PyCoreConfig *config, _PyCmdline *cmdline,
_PyOS_ResetGetOpt(); _PyOS_ResetGetOpt();
do { do {
int longindex = -1; int longindex = -1;
int c = _PyOS_GetOpt(cmdline->args->argc, cmdline->argv, PROGRAM_OPTS, int c = _PyOS_GetOpt(cmdline->args->argc, cmdline->argv, &longindex);
longoptions, &longindex);
if (c == EOF) { if (c == EOF) {
break; break;
} }
...@@ -1915,8 +1940,9 @@ config_parse_cmdline(_PyCoreConfig *config, _PyCmdline *cmdline, ...@@ -1915,8 +1940,9 @@ config_parse_cmdline(_PyCoreConfig *config, _PyCmdline *cmdline,
config->interactive++; config->interactive++;
break; break;
case 'E':
case 'I': case 'I':
config->preconfig.isolated++; /* option handled by _PyPreConfig_ReadFromArgv() */
break; break;
/* case 'J': reserved for Jython */ /* case 'J': reserved for Jython */
...@@ -1937,10 +1963,6 @@ config_parse_cmdline(_PyCoreConfig *config, _PyCmdline *cmdline, ...@@ -1937,10 +1963,6 @@ config_parse_cmdline(_PyCoreConfig *config, _PyCmdline *cmdline,
config->site_import = 0; config->site_import = 0;
break; break;
case 'E':
config->preconfig.use_environment = 0;
break;
case 't': case 't':
/* ignored for backwards compatibility */ /* ignored for backwards compatibility */
break; break;
...@@ -2235,7 +2257,8 @@ config_usage(int error, const wchar_t* program) ...@@ -2235,7 +2257,8 @@ config_usage(int error, const wchar_t* program)
/* Parse command line options and environment variables. */ /* Parse command line options and environment variables. */
static _PyInitError static _PyInitError
config_from_cmdline(_PyCoreConfig *config, _PyCmdline *cmdline) config_from_cmdline(_PyCoreConfig *config, _PyCmdline *cmdline,
const _PyPreConfig *preconfig)
{ {
int need_usage = 0; int need_usage = 0;
_PyInitError err; _PyInitError err;
...@@ -2271,7 +2294,7 @@ config_from_cmdline(_PyCoreConfig *config, _PyCmdline *cmdline) ...@@ -2271,7 +2294,7 @@ config_from_cmdline(_PyCoreConfig *config, _PyCmdline *cmdline)
return err; return err;
} }
err = _PyCoreConfig_Read(config); err = _PyCoreConfig_Read(config, preconfig);
if (_Py_INIT_FAILED(err)) { if (_Py_INIT_FAILED(err)) {
return err; return err;
} }
...@@ -2296,7 +2319,8 @@ config_from_cmdline(_PyCoreConfig *config, _PyCmdline *cmdline) ...@@ -2296,7 +2319,8 @@ config_from_cmdline(_PyCoreConfig *config, _PyCmdline *cmdline)
static _PyInitError static _PyInitError
config_read_from_argv_impl(_PyCoreConfig *config, const _PyArgv *args) config_read_from_argv_impl(_PyCoreConfig *config, const _PyArgv *args,
const _PyPreConfig *preconfig)
{ {
_PyInitError err; _PyInitError err;
...@@ -2309,7 +2333,7 @@ config_read_from_argv_impl(_PyCoreConfig *config, const _PyArgv *args) ...@@ -2309,7 +2333,7 @@ config_read_from_argv_impl(_PyCoreConfig *config, const _PyArgv *args)
goto done; goto done;
} }
err = config_from_cmdline(config, &cmdline); err = config_from_cmdline(config, &cmdline, preconfig);
if (_Py_INIT_FAILED(err)) { if (_Py_INIT_FAILED(err)) {
goto done; goto done;
} }
...@@ -2330,7 +2354,8 @@ done: ...@@ -2330,7 +2354,8 @@ done:
* Environment variables * Environment variables
* Py_xxx global configuration variables */ * Py_xxx global configuration variables */
_PyInitError _PyInitError
_PyCoreConfig_ReadFromArgv(_PyCoreConfig *config, const _PyArgv *args) _PyCoreConfig_ReadFromArgv(_PyCoreConfig *config, const _PyArgv *args,
const _PyPreConfig *preconfig)
{ {
_PyInitError err; _PyInitError err;
int init_utf8_mode = Py_UTF8Mode; int init_utf8_mode = Py_UTF8Mode;
...@@ -2381,7 +2406,7 @@ _PyCoreConfig_ReadFromArgv(_PyCoreConfig *config, const _PyArgv *args) ...@@ -2381,7 +2406,7 @@ _PyCoreConfig_ReadFromArgv(_PyCoreConfig *config, const _PyArgv *args)
Py_LegacyWindowsFSEncodingFlag = config->legacy_windows_fs_encoding; Py_LegacyWindowsFSEncodingFlag = config->legacy_windows_fs_encoding;
#endif #endif
err = config_read_from_argv_impl(config, args); err = config_read_from_argv_impl(config, args, preconfig);
if (_Py_INIT_FAILED(err)) { if (_Py_INIT_FAILED(err)) {
goto done; goto done;
} }
......
...@@ -43,6 +43,16 @@ wchar_t *_PyOS_optarg = NULL; /* optional argument */ ...@@ -43,6 +43,16 @@ wchar_t *_PyOS_optarg = NULL; /* optional argument */
static wchar_t *opt_ptr = L""; static wchar_t *opt_ptr = L"";
/* Python command line short and long options */
#define SHORT_OPTS L"bBc:dEhiIJm:OqRsStuvVW:xX:?"
static const _PyOS_LongOption longopts[] = {
{L"check-hash-based-pycs", 1, 0},
{NULL, 0, 0},
};
void _PyOS_ResetGetOpt(void) void _PyOS_ResetGetOpt(void)
{ {
_PyOS_opterr = 1; _PyOS_opterr = 1;
...@@ -51,8 +61,7 @@ void _PyOS_ResetGetOpt(void) ...@@ -51,8 +61,7 @@ void _PyOS_ResetGetOpt(void)
opt_ptr = L""; opt_ptr = L"";
} }
int _PyOS_GetOpt(int argc, wchar_t **argv, wchar_t *optstring, int _PyOS_GetOpt(int argc, wchar_t **argv, int *longindex)
const _PyOS_LongOption *longopts, int *longindex)
{ {
wchar_t *ptr; wchar_t *ptr;
wchar_t option; wchar_t option;
...@@ -128,7 +137,7 @@ int _PyOS_GetOpt(int argc, wchar_t **argv, wchar_t *optstring, ...@@ -128,7 +137,7 @@ int _PyOS_GetOpt(int argc, wchar_t **argv, wchar_t *optstring,
return '_'; return '_';
} }
if ((ptr = wcschr(optstring, option)) == NULL) { if ((ptr = wcschr(SHORT_OPTS, option)) == NULL) {
if (_PyOS_opterr) if (_PyOS_opterr)
fprintf(stderr, "Unknown option: -%c\n", (char)option); fprintf(stderr, "Unknown option: -%c\n", (char)option);
return '_'; return '_';
......
...@@ -393,7 +393,7 @@ pathconfig_global_init(void) ...@@ -393,7 +393,7 @@ pathconfig_global_init(void)
_PyInitError err; _PyInitError err;
_PyCoreConfig config = _PyCoreConfig_INIT; _PyCoreConfig config = _PyCoreConfig_INIT;
err = _PyCoreConfig_Read(&config); err = _PyCoreConfig_Read(&config, NULL);
if (_Py_INIT_FAILED(err)) { if (_Py_INIT_FAILED(err)) {
goto error; goto error;
} }
......
#include "Python.h" #include "Python.h"
#include "pycore_coreconfig.h" #include "pycore_coreconfig.h"
#include "pycore_getopt.h"
#define DECODE_LOCALE_ERR(NAME, LEN) \ #define DECODE_LOCALE_ERR(NAME, LEN) \
...@@ -92,6 +93,25 @@ _PyArgv_Decode(const _PyArgv *args, wchar_t*** argv_p) ...@@ -92,6 +93,25 @@ _PyArgv_Decode(const _PyArgv *args, wchar_t*** argv_p)
} }
/* --- _PyPreCmdline ------------------------------------------------- */
typedef struct {
const _PyArgv *args;
int argc;
wchar_t **argv;
} _PyPreCmdline;
static void
precmdline_clear(_PyPreCmdline *cmdline)
{
if (cmdline->args->use_bytes_argv && cmdline->argv != NULL) {
_Py_wstrlist_clear(cmdline->args->argc, cmdline->argv);
}
cmdline->argv = NULL;
}
/* --- _PyPreConfig ----------------------------------------------- */ /* --- _PyPreConfig ----------------------------------------------- */
void void
...@@ -169,6 +189,7 @@ _PyPreConfig_Read(_PyPreConfig *config) ...@@ -169,6 +189,7 @@ _PyPreConfig_Read(_PyPreConfig *config)
config->use_environment = 0; config->use_environment = 0;
} }
assert(config->isolated >= 0);
assert(config->use_environment >= 0); assert(config->use_environment >= 0);
return _Py_INIT_OK(); return _Py_INIT_OK();
...@@ -203,3 +224,76 @@ fail: ...@@ -203,3 +224,76 @@ fail:
#undef SET_ITEM #undef SET_ITEM
#undef SET_ITEM_INT #undef SET_ITEM_INT
} }
/* Parse the command line arguments */
static _PyInitError
preconfig_parse_cmdline(_PyPreConfig *config, _PyPreCmdline *cmdline)
{
_PyOS_ResetGetOpt();
/* Don't log parsing errors into stderr here: _PyCoreConfig_ReadFromArgv()
is responsible for that */
_PyOS_opterr = 0;
do {
int longindex = -1;
int c = _PyOS_GetOpt(cmdline->args->argc, cmdline->argv, &longindex);
if (c == EOF || c == 'c' || c == 'm') {
break;
}
switch (c) {
case 'E':
config->use_environment = 0;
break;
case 'I':
config->isolated++;
break;
default:
/* ignore other argument:
handled by _PyCoreConfig_ReadFromArgv() */
break;
}
} while (1);
return _Py_INIT_OK();
}
_PyInitError
_PyPreConfig_ReadFromArgv(_PyPreConfig *config, const _PyArgv *args)
{
_PyInitError err;
_PyPreCmdline cmdline;
memset(&cmdline, 0, sizeof(cmdline));
cmdline.args = args;
err = _PyArgv_Decode(cmdline.args, &cmdline.argv);
if (_Py_INIT_FAILED(err)) {
goto done;
}
err = preconfig_parse_cmdline(config, &cmdline);
if (_Py_INIT_FAILED(err)) {
goto done;
}
err = _PyPreConfig_Read(config);
if (_Py_INIT_FAILED(err)) {
goto done;
}
err = _Py_INIT_OK();
done:
precmdline_clear(&cmdline);
return err;
}
void
_PyPreConfig_Write(const _PyPreConfig *config)
{
}
...@@ -763,7 +763,7 @@ _Py_InitializeCore(PyInterpreterState **interp_p, ...@@ -763,7 +763,7 @@ _Py_InitializeCore(PyInterpreterState **interp_p,
_PyMem_SetDefaultAllocator(PYMEM_DOMAIN_RAW, &old_alloc); _PyMem_SetDefaultAllocator(PYMEM_DOMAIN_RAW, &old_alloc);
if (_PyCoreConfig_Copy(&config, src_config) >= 0) { if (_PyCoreConfig_Copy(&config, src_config) >= 0) {
err = _PyCoreConfig_Read(&config); err = _PyCoreConfig_Read(&config, NULL);
} }
else { else {
err = _Py_INIT_ERR("failed to copy core config"); err = _Py_INIT_ERR("failed to copy core config");
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment