0
0
mirror of https://github.com/phpv8/v8js.git synced 2025-01-08 22:11:52 +00:00

Fix a bunch of memory leaks.

Found by compiling PHP with --enable-debug.
This commit is contained in:
C. Scott Ananian 2013-10-26 01:20:41 -04:00
parent 7cdcb5f45e
commit b4d560dd52
2 changed files with 22 additions and 7 deletions

View File

@ -512,6 +512,10 @@ static void php_v8js_free_storage(void *object TSRMLS_DC) /* {{{ */
zval_ptr_dtor(&c->pending_exception); zval_ptr_dtor(&c->pending_exception);
} }
if (c->module_loader) {
zval_ptr_dtor(&c->module_loader);
}
/* Delete PHP global object from JavaScript */ /* Delete PHP global object from JavaScript */
if (!c->context.IsEmpty()) { if (!c->context.IsEmpty()) {
v8::Locker locker(c->isolate); v8::Locker locker(c->isolate);

View File

@ -234,6 +234,7 @@ V8JS_METHOD(require)
for (std::vector<char *>::iterator it = c->modules_stack.begin(); it != c->modules_stack.end(); ++it) for (std::vector<char *>::iterator it = c->modules_stack.begin(); it != c->modules_stack.end(); ++it)
{ {
if (!strcmp(*it, normalised_module_id)) { if (!strcmp(*it, normalised_module_id)) {
efree(normalised_module_id);
efree(normalised_path); efree(normalised_path);
info.GetReturnValue().Set(v8::ThrowException(V8JS_SYM("Module cyclic dependency"))); info.GetReturnValue().Set(v8::ThrowException(V8JS_SYM("Module cyclic dependency")));
@ -243,6 +244,7 @@ V8JS_METHOD(require)
// If we have already loaded and cached this module then use it // If we have already loaded and cached this module then use it
if (V8JSG(modules_loaded).count(normalised_module_id) > 0) { if (V8JSG(modules_loaded).count(normalised_module_id) > 0) {
efree(normalised_module_id);
efree(normalised_path); efree(normalised_path);
info.GetReturnValue().Set(V8JSG(modules_loaded)[normalised_module_id]); info.GetReturnValue().Set(V8JSG(modules_loaded)[normalised_module_id]);
@ -251,22 +253,26 @@ V8JS_METHOD(require)
// Callback to PHP to load the module code // Callback to PHP to load the module code
zval module_code; zval *module_code;
zval *normalised_path_zend; zval *normalised_path_zend;
MAKE_STD_ZVAL(normalised_path_zend); MAKE_STD_ZVAL(normalised_path_zend);
ZVAL_STRING(normalised_path_zend, normalised_module_id, 1); ZVAL_STRING(normalised_path_zend, normalised_module_id, 1);
zval* params[] = { normalised_path_zend };
if (FAILURE == call_user_function(EG(function_table), NULL, c->module_loader, &module_code, 1, params TSRMLS_CC)) { zval **params[1] = {&normalised_path_zend};
if (FAILURE == call_user_function_ex(EG(function_table), NULL, c->module_loader, &module_code, 1, params, 0, NULL TSRMLS_CC)) {
zval_ptr_dtor(&normalised_path_zend);
efree(normalised_module_id);
efree(normalised_path); efree(normalised_path);
info.GetReturnValue().Set(v8::ThrowException(V8JS_SYM("Module loader callback failed"))); info.GetReturnValue().Set(v8::ThrowException(V8JS_SYM("Module loader callback failed")));
return; return;
} }
zval_ptr_dtor(&normalised_path_zend);
// Check if an exception was thrown // Check if an exception was thrown
if (EG(exception)) { if (EG(exception)) {
efree(normalised_module_id);
efree(normalised_path); efree(normalised_path);
// Clear the PHP exception and throw it in V8 instead // Clear the PHP exception and throw it in V8 instead
@ -276,12 +282,14 @@ V8JS_METHOD(require)
} }
// Convert the return value to string // Convert the return value to string
if (Z_TYPE(module_code) != IS_STRING) { if (Z_TYPE_P(module_code) != IS_STRING) {
convert_to_string(&module_code); convert_to_string(module_code);
} }
// Check that some code has been returned // Check that some code has been returned
if (!strlen(Z_STRVAL(module_code))) { if (Z_STRLEN_P(module_code)==0) {
zval_ptr_dtor(&module_code);
efree(normalised_module_id);
efree(normalised_path); efree(normalised_path);
info.GetReturnValue().Set(v8::ThrowException(V8JS_SYM("Module loader callback did not return code"))); info.GetReturnValue().Set(v8::ThrowException(V8JS_SYM("Module loader callback did not return code")));
@ -321,13 +329,15 @@ V8JS_METHOD(require)
// Set script identifier // Set script identifier
v8::Local<v8::String> sname = V8JS_SYM("require"); v8::Local<v8::String> sname = V8JS_SYM("require");
v8::Local<v8::String> source = V8JS_STR(Z_STRVAL(module_code)); v8::Local<v8::String> source = V8JS_STRL(Z_STRVAL_P(module_code), Z_STRLEN_P(module_code));
zval_ptr_dtor(&module_code);
// Create and compile script // Create and compile script
v8::Local<v8::Script> script = v8::Script::New(source, sname); v8::Local<v8::Script> script = v8::Script::New(source, sname);
// The script will be empty if there are compile errors // The script will be empty if there are compile errors
if (script.IsEmpty()) { if (script.IsEmpty()) {
efree(normalised_module_id);
efree(normalised_path); efree(normalised_path);
info.GetReturnValue().Set(v8::ThrowException(V8JS_SYM("Module script compile failed"))); info.GetReturnValue().Set(v8::ThrowException(V8JS_SYM("Module script compile failed")));
return; return;
@ -345,6 +355,7 @@ V8JS_METHOD(require)
c->modules_stack.pop_back(); c->modules_stack.pop_back();
c->modules_base.pop_back(); c->modules_base.pop_back();
efree(normalised_module_id);
efree(normalised_path); efree(normalised_path);
// Script possibly terminated, return immediately // Script possibly terminated, return immediately