Commit b16b4e45 authored by Victor Stinner's avatar Victor Stinner Committed by GitHub

bpo-36763: Add PyMemAllocatorName (GH-13387)

* Add PyMemAllocatorName enum
* _PyPreConfig.allocator type becomes PyMemAllocatorName, instead of
  char*
* Remove _PyPreConfig_Clear()
* Add _PyMem_GetAllocatorName()
* Rename _PyMem_GetAllocatorsName() to
  _PyMem_GetCurrentAllocatorName()
* Remove _PyPreConfig_SetAllocator(): just call
  _PyMem_SetupAllocators() directly, we don't have do reallocate the
  configuration with the new allocator anymore!
* _PyPreConfig_Write() parameter becomes const, as it should be in
  the first place!
parent 80ed3533
...@@ -120,7 +120,9 @@ typedef struct { ...@@ -120,7 +120,9 @@ typedef struct {
int utf8_mode; int utf8_mode;
int dev_mode; /* Development mode. PYTHONDEVMODE, -X dev */ int dev_mode; /* Development mode. PYTHONDEVMODE, -X dev */
char *allocator; /* Memory allocator: PYTHONMALLOC */
/* Memory allocator: PYTHONMALLOC env var */
PyMemAllocatorName allocator;
} _PyPreConfig; } _PyPreConfig;
#ifdef MS_WINDOWS #ifdef MS_WINDOWS
...@@ -137,7 +139,7 @@ typedef struct { ...@@ -137,7 +139,7 @@ typedef struct {
.isolated = -1, \ .isolated = -1, \
.use_environment = -1, \ .use_environment = -1, \
.dev_mode = -1, \ .dev_mode = -1, \
.allocator = NULL} .allocator = PYMEM_ALLOCATOR_NOT_SET}
/* --- _PyCoreConfig ---------------------------------------------- */ /* --- _PyCoreConfig ---------------------------------------------- */
......
...@@ -11,12 +11,8 @@ PyAPI_FUNC(void *) PyMem_RawCalloc(size_t nelem, size_t elsize); ...@@ -11,12 +11,8 @@ PyAPI_FUNC(void *) PyMem_RawCalloc(size_t nelem, size_t elsize);
PyAPI_FUNC(void *) PyMem_RawRealloc(void *ptr, size_t new_size); PyAPI_FUNC(void *) PyMem_RawRealloc(void *ptr, size_t new_size);
PyAPI_FUNC(void) PyMem_RawFree(void *ptr); PyAPI_FUNC(void) PyMem_RawFree(void *ptr);
/* Configure the Python memory allocators. Pass NULL to use default
allocators. */
PyAPI_FUNC(int) _PyMem_SetupAllocators(const char *opt);
/* Try to get the allocators name set by _PyMem_SetupAllocators(). */ /* Try to get the allocators name set by _PyMem_SetupAllocators(). */
PyAPI_FUNC(const char*) _PyMem_GetAllocatorsName(void); PyAPI_FUNC(const char*) _PyMem_GetCurrentAllocatorName(void);
PyAPI_FUNC(void *) PyMem_Calloc(size_t nelem, size_t elsize); PyAPI_FUNC(void *) PyMem_Calloc(size_t nelem, size_t elsize);
...@@ -41,6 +37,19 @@ typedef enum { ...@@ -41,6 +37,19 @@ typedef enum {
PYMEM_DOMAIN_OBJ PYMEM_DOMAIN_OBJ
} PyMemAllocatorDomain; } PyMemAllocatorDomain;
typedef enum {
PYMEM_ALLOCATOR_NOT_SET = 0,
PYMEM_ALLOCATOR_DEFAULT = 1,
PYMEM_ALLOCATOR_DEBUG = 2,
PYMEM_ALLOCATOR_MALLOC = 3,
PYMEM_ALLOCATOR_MALLOC_DEBUG = 4,
#ifdef WITH_PYMALLOC
PYMEM_ALLOCATOR_PYMALLOC = 5,
PYMEM_ALLOCATOR_PYMALLOC_DEBUG = 6,
#endif
} PyMemAllocatorName;
typedef struct { typedef struct {
/* user context passed as the first argument to the 4 functions */ /* user context passed as the first argument to the 4 functions */
void *ctx; void *ctx;
......
...@@ -88,7 +88,6 @@ PyAPI_FUNC(_PyInitError) _PyPreCmdline_Read(_PyPreCmdline *cmdline, ...@@ -88,7 +88,6 @@ PyAPI_FUNC(_PyInitError) _PyPreCmdline_Read(_PyPreCmdline *cmdline,
/* --- _PyPreConfig ----------------------------------------------- */ /* --- _PyPreConfig ----------------------------------------------- */
PyAPI_FUNC(void) _PyPreConfig_Clear(_PyPreConfig *config);
PyAPI_FUNC(int) _PyPreConfig_Copy(_PyPreConfig *config, PyAPI_FUNC(int) _PyPreConfig_Copy(_PyPreConfig *config,
const _PyPreConfig *config2); const _PyPreConfig *config2);
PyAPI_FUNC(PyObject*) _PyPreConfig_AsDict(const _PyPreConfig *config); PyAPI_FUNC(PyObject*) _PyPreConfig_AsDict(const _PyPreConfig *config);
...@@ -96,7 +95,7 @@ PyAPI_FUNC(void) _PyCoreConfig_GetCoreConfig(_PyPreConfig *config, ...@@ -96,7 +95,7 @@ PyAPI_FUNC(void) _PyCoreConfig_GetCoreConfig(_PyPreConfig *config,
const _PyCoreConfig *core_config); const _PyCoreConfig *core_config);
PyAPI_FUNC(_PyInitError) _PyPreConfig_Read(_PyPreConfig *config, PyAPI_FUNC(_PyInitError) _PyPreConfig_Read(_PyPreConfig *config,
const _PyArgv *args); const _PyArgv *args);
PyAPI_FUNC(_PyInitError) _PyPreConfig_Write(_PyPreConfig *config); PyAPI_FUNC(_PyInitError) _PyPreConfig_Write(const _PyPreConfig *config);
/* --- _PyCoreConfig ---------------------------------------------- */ /* --- _PyCoreConfig ---------------------------------------------- */
......
...@@ -179,6 +179,15 @@ static inline int _PyMem_IsPtrFreed(void *ptr) ...@@ -179,6 +179,15 @@ static inline int _PyMem_IsPtrFreed(void *ptr)
#endif #endif
} }
PyAPI_FUNC(int) _PyMem_GetAllocatorName(
const char *name,
PyMemAllocatorName *allocator);
/* Configure the Python memory allocators.
Pass PYMEM_ALLOCATOR_DEFAULT to use default allocators.
PYMEM_ALLOCATOR_NOT_SET does nothing. */
PyAPI_FUNC(int) _PyMem_SetupAllocators(PyMemAllocatorName allocator);
#ifdef __cplusplus #ifdef __cplusplus
} }
#endif #endif
......
...@@ -13,6 +13,9 @@ import textwrap ...@@ -13,6 +13,9 @@ import textwrap
MS_WINDOWS = (os.name == 'nt') MS_WINDOWS = (os.name == 'nt')
PYMEM_ALLOCATOR_NOT_SET = 0
PYMEM_ALLOCATOR_DEBUG = 2
PYMEM_ALLOCATOR_MALLOC = 3
class EmbeddingTestsMixin: class EmbeddingTestsMixin:
...@@ -272,7 +275,7 @@ class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase): ...@@ -272,7 +275,7 @@ class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase):
# Mark config which should be get by get_default_config() # Mark config which should be get by get_default_config()
GET_DEFAULT_CONFIG = object() GET_DEFAULT_CONFIG = object()
DEFAULT_PRE_CONFIG = { DEFAULT_PRE_CONFIG = {
'allocator': None, 'allocator': PYMEM_ALLOCATOR_NOT_SET,
'coerce_c_locale': 0, 'coerce_c_locale': 0,
'coerce_c_locale_warn': 0, 'coerce_c_locale_warn': 0,
'utf8_mode': 0, 'utf8_mode': 0,
...@@ -564,7 +567,7 @@ class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase): ...@@ -564,7 +567,7 @@ class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase):
def test_init_from_config(self): def test_init_from_config(self):
preconfig = { preconfig = {
'allocator': 'malloc', 'allocator': PYMEM_ALLOCATOR_MALLOC,
'utf8_mode': 1, 'utf8_mode': 1,
} }
config = { config = {
...@@ -608,7 +611,7 @@ class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase): ...@@ -608,7 +611,7 @@ class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase):
self.check_config("init_from_config", config, preconfig) self.check_config("init_from_config", config, preconfig)
INIT_ENV_PRECONFIG = { INIT_ENV_PRECONFIG = {
'allocator': 'malloc', 'allocator': PYMEM_ALLOCATOR_MALLOC,
} }
INIT_ENV_CONFIG = { INIT_ENV_CONFIG = {
'use_hash_seed': 1, 'use_hash_seed': 1,
...@@ -633,7 +636,7 @@ class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase): ...@@ -633,7 +636,7 @@ class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase):
def test_init_env_dev_mode(self): def test_init_env_dev_mode(self):
preconfig = dict(self.INIT_ENV_PRECONFIG, preconfig = dict(self.INIT_ENV_PRECONFIG,
allocator='debug') allocator=PYMEM_ALLOCATOR_DEBUG)
config = dict(self.INIT_ENV_CONFIG, config = dict(self.INIT_ENV_CONFIG,
dev_mode=1, dev_mode=1,
warnoptions=['default']) warnoptions=['default'])
...@@ -641,7 +644,7 @@ class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase): ...@@ -641,7 +644,7 @@ class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase):
def test_init_env_dev_mode_alloc(self): def test_init_env_dev_mode_alloc(self):
preconfig = dict(self.INIT_ENV_PRECONFIG, preconfig = dict(self.INIT_ENV_PRECONFIG,
allocator='malloc') allocator=PYMEM_ALLOCATOR_MALLOC)
config = dict(self.INIT_ENV_CONFIG, config = dict(self.INIT_ENV_CONFIG,
dev_mode=1, dev_mode=1,
warnoptions=['default']) warnoptions=['default'])
...@@ -649,7 +652,7 @@ class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase): ...@@ -649,7 +652,7 @@ class InitConfigTests(EmbeddingTestsMixin, unittest.TestCase):
def test_init_dev_mode(self): def test_init_dev_mode(self):
preconfig = { preconfig = {
'allocator': 'debug', 'allocator': PYMEM_ALLOCATOR_DEBUG,
} }
config = { config = {
'faulthandler': 1, 'faulthandler': 1,
......
...@@ -4297,7 +4297,7 @@ pymem_malloc_without_gil(PyObject *self, PyObject *args) ...@@ -4297,7 +4297,7 @@ pymem_malloc_without_gil(PyObject *self, PyObject *args)
static PyObject* static PyObject*
test_pymem_getallocatorsname(PyObject *self, PyObject *args) test_pymem_getallocatorsname(PyObject *self, PyObject *args)
{ {
const char *name = _PyMem_GetAllocatorsName(); const char *name = _PyMem_GetCurrentAllocatorName();
if (name == NULL) { if (name == NULL) {
PyErr_SetString(PyExc_RuntimeError, "cannot get allocators name"); PyErr_SetString(PyExc_RuntimeError, "cannot get allocators name");
return NULL; return NULL;
......
...@@ -268,26 +268,65 @@ _PyMem_SetDefaultAllocator(PyMemAllocatorDomain domain, ...@@ -268,26 +268,65 @@ _PyMem_SetDefaultAllocator(PyMemAllocatorDomain domain,
int int
_PyMem_SetupAllocators(const char *opt) _PyMem_GetAllocatorName(const char *name, PyMemAllocatorName *allocator)
{ {
if (opt == NULL || *opt == '\0') { if (name == NULL || *name == '\0') {
/* PYTHONMALLOC is empty or is not set or ignored (-E/-I command line /* PYTHONMALLOC is empty or is not set or ignored (-E/-I command line
options): use default memory allocators */ nameions): use default memory allocators */
opt = "default"; *allocator = PYMEM_ALLOCATOR_DEFAULT;
} }
else if (strcmp(name, "default") == 0) {
*allocator = PYMEM_ALLOCATOR_DEFAULT;
}
else if (strcmp(name, "debug") == 0) {
*allocator = PYMEM_ALLOCATOR_DEBUG;
}
#ifdef WITH_PYMALLOC
else if (strcmp(name, "pymalloc") == 0) {
*allocator = PYMEM_ALLOCATOR_PYMALLOC;
}
else if (strcmp(name, "pymalloc_debug") == 0) {
*allocator = PYMEM_ALLOCATOR_PYMALLOC_DEBUG;
}
#endif
else if (strcmp(name, "malloc") == 0) {
*allocator = PYMEM_ALLOCATOR_MALLOC;
}
else if (strcmp(name, "malloc_debug") == 0) {
*allocator = PYMEM_ALLOCATOR_MALLOC_DEBUG;
}
else {
/* unknown allocator */
return -1;
}
return 0;
}
if (strcmp(opt, "default") == 0) { int
_PyMem_SetupAllocators(PyMemAllocatorName allocator)
{
switch (allocator) {
case PYMEM_ALLOCATOR_NOT_SET:
/* do nothing */
break;
case PYMEM_ALLOCATOR_DEFAULT:
(void)_PyMem_SetDefaultAllocator(PYMEM_DOMAIN_RAW, NULL); (void)_PyMem_SetDefaultAllocator(PYMEM_DOMAIN_RAW, NULL);
(void)_PyMem_SetDefaultAllocator(PYMEM_DOMAIN_MEM, NULL); (void)_PyMem_SetDefaultAllocator(PYMEM_DOMAIN_MEM, NULL);
(void)_PyMem_SetDefaultAllocator(PYMEM_DOMAIN_OBJ, NULL); (void)_PyMem_SetDefaultAllocator(PYMEM_DOMAIN_OBJ, NULL);
} break;
else if (strcmp(opt, "debug") == 0) {
case PYMEM_ALLOCATOR_DEBUG:
(void)pymem_set_default_allocator(PYMEM_DOMAIN_RAW, 1, NULL); (void)pymem_set_default_allocator(PYMEM_DOMAIN_RAW, 1, NULL);
(void)pymem_set_default_allocator(PYMEM_DOMAIN_MEM, 1, NULL); (void)pymem_set_default_allocator(PYMEM_DOMAIN_MEM, 1, NULL);
(void)pymem_set_default_allocator(PYMEM_DOMAIN_OBJ, 1, NULL); (void)pymem_set_default_allocator(PYMEM_DOMAIN_OBJ, 1, NULL);
} break;
#ifdef WITH_PYMALLOC #ifdef WITH_PYMALLOC
else if (strcmp(opt, "pymalloc") == 0 || strcmp(opt, "pymalloc_debug") == 0) { case PYMEM_ALLOCATOR_PYMALLOC:
case PYMEM_ALLOCATOR_PYMALLOC_DEBUG:
{
PyMemAllocatorEx malloc_alloc = MALLOC_ALLOC; PyMemAllocatorEx malloc_alloc = MALLOC_ALLOC;
PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &malloc_alloc); PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &malloc_alloc);
...@@ -295,22 +334,28 @@ _PyMem_SetupAllocators(const char *opt) ...@@ -295,22 +334,28 @@ _PyMem_SetupAllocators(const char *opt)
PyMem_SetAllocator(PYMEM_DOMAIN_MEM, &pymalloc); PyMem_SetAllocator(PYMEM_DOMAIN_MEM, &pymalloc);
PyMem_SetAllocator(PYMEM_DOMAIN_OBJ, &pymalloc); PyMem_SetAllocator(PYMEM_DOMAIN_OBJ, &pymalloc);
if (strcmp(opt, "pymalloc_debug") == 0) { if (allocator == PYMEM_ALLOCATOR_PYMALLOC_DEBUG) {
PyMem_SetupDebugHooks(); PyMem_SetupDebugHooks();
} }
break;
} }
#endif #endif
else if (strcmp(opt, "malloc") == 0 || strcmp(opt, "malloc_debug") == 0) {
case PYMEM_ALLOCATOR_MALLOC:
case PYMEM_ALLOCATOR_MALLOC_DEBUG:
{
PyMemAllocatorEx malloc_alloc = MALLOC_ALLOC; PyMemAllocatorEx malloc_alloc = MALLOC_ALLOC;
PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &malloc_alloc); PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &malloc_alloc);
PyMem_SetAllocator(PYMEM_DOMAIN_MEM, &malloc_alloc); PyMem_SetAllocator(PYMEM_DOMAIN_MEM, &malloc_alloc);
PyMem_SetAllocator(PYMEM_DOMAIN_OBJ, &malloc_alloc); PyMem_SetAllocator(PYMEM_DOMAIN_OBJ, &malloc_alloc);
if (strcmp(opt, "malloc_debug") == 0) { if (allocator == PYMEM_ALLOCATOR_MALLOC_DEBUG) {
PyMem_SetupDebugHooks(); PyMem_SetupDebugHooks();
} }
break;
} }
else {
default:
/* unknown allocator */ /* unknown allocator */
return -1; return -1;
} }
...@@ -326,7 +371,7 @@ pymemallocator_eq(PyMemAllocatorEx *a, PyMemAllocatorEx *b) ...@@ -326,7 +371,7 @@ pymemallocator_eq(PyMemAllocatorEx *a, PyMemAllocatorEx *b)
const char* const char*
_PyMem_GetAllocatorsName(void) _PyMem_GetCurrentAllocatorName(void)
{ {
PyMemAllocatorEx malloc_alloc = MALLOC_ALLOC; PyMemAllocatorEx malloc_alloc = MALLOC_ALLOC;
#ifdef WITH_PYMALLOC #ifdef WITH_PYMALLOC
......
...@@ -379,7 +379,7 @@ static int test_init_from_config(void) ...@@ -379,7 +379,7 @@ static int test_init_from_config(void)
_PyPreConfig preconfig = _PyPreConfig_INIT; _PyPreConfig preconfig = _PyPreConfig_INIT;
putenv("PYTHONMALLOC=malloc_debug"); putenv("PYTHONMALLOC=malloc_debug");
preconfig.allocator = "malloc"; preconfig.allocator = PYMEM_ALLOCATOR_MALLOC;
putenv("PYTHONUTF8=0"); putenv("PYTHONUTF8=0");
Py_UTF8Mode = 0; Py_UTF8Mode = 0;
......
...@@ -2021,26 +2021,22 @@ core_read_precmdline(_PyCoreConfig *config, _PyPreCmdline *precmdline) ...@@ -2021,26 +2021,22 @@ core_read_precmdline(_PyCoreConfig *config, _PyPreCmdline *precmdline)
_PyPreConfig preconfig = _PyPreConfig_INIT; _PyPreConfig preconfig = _PyPreConfig_INIT;
if (_PyPreConfig_Copy(&preconfig, &_PyRuntime.preconfig) < 0) { if (_PyPreConfig_Copy(&preconfig, &_PyRuntime.preconfig) < 0) {
err = _Py_INIT_NO_MEMORY(); err = _Py_INIT_NO_MEMORY();
goto done; return err;
} }
_PyCoreConfig_GetCoreConfig(&preconfig, config); _PyCoreConfig_GetCoreConfig(&preconfig, config);
err = _PyPreCmdline_Read(precmdline, &preconfig); err = _PyPreCmdline_Read(precmdline, &preconfig);
if (_Py_INIT_FAILED(err)) { if (_Py_INIT_FAILED(err)) {
goto done; return err;
} }
if (_PyPreCmdline_SetCoreConfig(precmdline, config) < 0) { if (_PyPreCmdline_SetCoreConfig(precmdline, config) < 0) {
err = _Py_INIT_NO_MEMORY(); err = _Py_INIT_NO_MEMORY();
goto done; return err;
} }
err = _Py_INIT_OK(); return _Py_INIT_OK();
done:
_PyPreConfig_Clear(&preconfig);
return err;
} }
......
...@@ -260,19 +260,9 @@ _PyPreCmdline_Read(_PyPreCmdline *cmdline, ...@@ -260,19 +260,9 @@ _PyPreCmdline_Read(_PyPreCmdline *cmdline,
/* --- _PyPreConfig ----------------------------------------------- */ /* --- _PyPreConfig ----------------------------------------------- */
void
_PyPreConfig_Clear(_PyPreConfig *config)
{
PyMem_RawFree(config->allocator);
config->allocator = NULL;
}
int int
_PyPreConfig_Copy(_PyPreConfig *config, const _PyPreConfig *config2) _PyPreConfig_Copy(_PyPreConfig *config, const _PyPreConfig *config2)
{ {
_PyPreConfig_Clear(config);
#define COPY_ATTR(ATTR) config->ATTR = config2->ATTR #define COPY_ATTR(ATTR) config->ATTR = config2->ATTR
#define COPY_STR_ATTR(ATTR) \ #define COPY_STR_ATTR(ATTR) \
do { \ do { \
...@@ -293,7 +283,7 @@ _PyPreConfig_Copy(_PyPreConfig *config, const _PyPreConfig *config2) ...@@ -293,7 +283,7 @@ _PyPreConfig_Copy(_PyPreConfig *config, const _PyPreConfig *config2)
COPY_ATTR(legacy_windows_fs_encoding); COPY_ATTR(legacy_windows_fs_encoding);
#endif #endif
COPY_ATTR(utf8_mode); COPY_ATTR(utf8_mode);
COPY_STR_ATTR(allocator); COPY_ATTR(allocator);
#undef COPY_ATTR #undef COPY_ATTR
#undef COPY_STR_ATTR #undef COPY_STR_ATTR
...@@ -341,7 +331,7 @@ _PyPreConfig_AsDict(const _PyPreConfig *config) ...@@ -341,7 +331,7 @@ _PyPreConfig_AsDict(const _PyPreConfig *config)
SET_ITEM_INT(legacy_windows_fs_encoding); SET_ITEM_INT(legacy_windows_fs_encoding);
#endif #endif
SET_ITEM_INT(dev_mode); SET_ITEM_INT(dev_mode);
SET_ITEM_STR(allocator); SET_ITEM_INT(allocator);
return dict; return dict;
fail: fail:
...@@ -616,25 +606,21 @@ preconfig_init_coerce_c_locale(_PyPreConfig *config) ...@@ -616,25 +606,21 @@ preconfig_init_coerce_c_locale(_PyPreConfig *config)
static _PyInitError static _PyInitError
preconfig_init_allocator(_PyPreConfig *config) preconfig_init_allocator(_PyPreConfig *config)
{ {
if (config->allocator == NULL) { if (config->allocator == PYMEM_ALLOCATOR_NOT_SET) {
/* bpo-34247. The PYTHONMALLOC environment variable has the priority /* bpo-34247. The PYTHONMALLOC environment variable has the priority
over PYTHONDEV env var and "-X dev" command line option. over PYTHONDEV env var and "-X dev" command line option.
For example, PYTHONMALLOC=malloc PYTHONDEVMODE=1 sets the memory For example, PYTHONMALLOC=malloc PYTHONDEVMODE=1 sets the memory
allocators to "malloc" (and not to "debug"). */ allocators to "malloc" (and not to "debug"). */
const char *allocator = _Py_GetEnv(config->use_environment, "PYTHONMALLOC"); const char *envvar = _Py_GetEnv(config->use_environment, "PYTHONMALLOC");
if (allocator) { if (envvar) {
config->allocator = _PyMem_RawStrdup(allocator); if (_PyMem_GetAllocatorName(envvar, &config->allocator) < 0) {
if (config->allocator == NULL) { return _Py_INIT_ERR("PYTHONMALLOC: unknown allocator");
return _Py_INIT_NO_MEMORY();
} }
} }
} }
if (config->dev_mode && config->allocator == NULL) { if (config->dev_mode && config->allocator == PYMEM_ALLOCATOR_NOT_SET) {
config->allocator = _PyMem_RawStrdup("debug"); config->allocator = PYMEM_ALLOCATOR_DEBUG;
if (config->allocator == NULL) {
return _Py_INIT_NO_MEMORY();
}
} }
return _Py_INIT_OK(); return _Py_INIT_OK();
} }
...@@ -815,7 +801,6 @@ done: ...@@ -815,7 +801,6 @@ done:
setlocale(LC_CTYPE, init_ctype_locale); setlocale(LC_CTYPE, init_ctype_locale);
PyMem_RawFree(init_ctype_locale); PyMem_RawFree(init_ctype_locale);
} }
_PyPreConfig_Clear(&save_config);
Py_UTF8Mode = init_utf8_mode ; Py_UTF8Mode = init_utf8_mode ;
#ifdef MS_WINDOWS #ifdef MS_WINDOWS
Py_LegacyWindowsFSEncodingFlag = init_legacy_encoding; Py_LegacyWindowsFSEncodingFlag = init_legacy_encoding;
...@@ -825,40 +810,6 @@ done: ...@@ -825,40 +810,6 @@ done:
} }
static _PyInitError
_PyPreConfig_SetAllocator(_PyPreConfig *config)
{
assert(!_PyRuntime.core_initialized);
PyMemAllocatorEx old_alloc;
PyMem_GetAllocator(PYMEM_DOMAIN_RAW, &old_alloc);
if (_PyMem_SetupAllocators(config->allocator) < 0) {
return _Py_INIT_ERR("Unknown PYTHONMALLOC allocator");
}
/* Copy the pre-configuration with the new allocator */
_PyPreConfig config2 = _PyPreConfig_INIT;
if (_PyPreConfig_Copy(&config2, config) < 0) {
_PyPreConfig_Clear(&config2);
PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &old_alloc);
return _Py_INIT_NO_MEMORY();
}
/* Free the old config and replace config with config2. Since config now
owns the data, don't free config2. */
PyMemAllocatorEx new_alloc;
PyMem_GetAllocator(PYMEM_DOMAIN_RAW, &new_alloc);
PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &old_alloc);
_PyPreConfig_Clear(config);
PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &new_alloc);
*config = config2;
return _Py_INIT_OK();
}
/* Write the pre-configuration: /* Write the pre-configuration:
- set the memory allocators - set the memory allocators
...@@ -872,7 +823,7 @@ _PyPreConfig_SetAllocator(_PyPreConfig *config) ...@@ -872,7 +823,7 @@ _PyPreConfig_SetAllocator(_PyPreConfig *config)
Do nothing if called after Py_Initialize(): ignore the new Do nothing if called after Py_Initialize(): ignore the new
pre-configuration. */ pre-configuration. */
_PyInitError _PyInitError
_PyPreConfig_Write(_PyPreConfig *config) _PyPreConfig_Write(const _PyPreConfig *config)
{ {
if (_PyRuntime.core_initialized) { if (_PyRuntime.core_initialized) {
/* bpo-34008: Calling this functions after Py_Initialize() ignores /* bpo-34008: Calling this functions after Py_Initialize() ignores
...@@ -880,10 +831,9 @@ _PyPreConfig_Write(_PyPreConfig *config) ...@@ -880,10 +831,9 @@ _PyPreConfig_Write(_PyPreConfig *config)
return _Py_INIT_OK(); return _Py_INIT_OK();
} }
if (config->allocator != NULL) { if (config->allocator != PYMEM_ALLOCATOR_NOT_SET) {
_PyInitError err = _PyPreConfig_SetAllocator(config); if (_PyMem_SetupAllocators(config->allocator) < 0) {
if (_Py_INIT_FAILED(err)) { return _Py_INIT_ERR("Unknown PYTHONMALLOC allocator");
return err;
} }
} }
......
...@@ -706,26 +706,22 @@ _Py_PreInitializeFromPyArgv(const _PyPreConfig *src_config, const _PyArgv *args) ...@@ -706,26 +706,22 @@ _Py_PreInitializeFromPyArgv(const _PyPreConfig *src_config, const _PyArgv *args)
if (src_config) { if (src_config) {
if (_PyPreConfig_Copy(&config, src_config) < 0) { if (_PyPreConfig_Copy(&config, src_config) < 0) {
err = _Py_INIT_NO_MEMORY(); err = _Py_INIT_NO_MEMORY();
goto done; return err;
} }
} }
err = _PyPreConfig_Read(&config, args); err = _PyPreConfig_Read(&config, args);
if (_Py_INIT_FAILED(err)) { if (_Py_INIT_FAILED(err)) {
goto done; return err;
} }
err = _PyPreConfig_Write(&config); err = _PyPreConfig_Write(&config);
if (_Py_INIT_FAILED(err)) { if (_Py_INIT_FAILED(err)) {
goto done; return err;
} }
runtime->pre_initialized = 1; runtime->pre_initialized = 1;
err = _Py_INIT_OK(); return _Py_INIT_OK();
done:
_PyPreConfig_Clear(&config);
return err;
} }
......
...@@ -106,8 +106,6 @@ _PyRuntimeState_Fini(_PyRuntimeState *runtime) ...@@ -106,8 +106,6 @@ _PyRuntimeState_Fini(_PyRuntimeState *runtime)
runtime->xidregistry.mutex = NULL; runtime->xidregistry.mutex = NULL;
} }
_PyPreConfig_Clear(&runtime->preconfig);
PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &old_alloc); PyMem_SetAllocator(PYMEM_DOMAIN_RAW, &old_alloc);
} }
......
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