mirror of
https://github.com/jerryscript-project/jerryscript.git
synced 2025-12-15 16:29:21 +00:00
234 lines
6.9 KiB
C
234 lines
6.9 KiB
C
/* Copyright JS Foundation and other contributors, http://js.foundation
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "jerryscript.h"
|
|
|
|
#include "config.h"
|
|
#include "test-common.h"
|
|
|
|
static int mode = 0;
|
|
static int counter = 0;
|
|
|
|
static void
|
|
vm_throw_callback (const jerry_value_t error_value, /**< captured error */
|
|
void *user_p) /**< user pointer */
|
|
{
|
|
TEST_ASSERT (user_p == (void *) &mode);
|
|
counter++;
|
|
|
|
switch (mode)
|
|
{
|
|
case 0:
|
|
{
|
|
TEST_ASSERT (counter == 1);
|
|
TEST_ASSERT (jerry_value_is_number (error_value) && jerry_get_number_value (error_value) == -5.6);
|
|
break;
|
|
}
|
|
case 1:
|
|
{
|
|
TEST_ASSERT (counter == 1);
|
|
TEST_ASSERT (jerry_value_is_null (error_value));
|
|
break;
|
|
}
|
|
case 2:
|
|
{
|
|
jerry_char_t string_buf[2];
|
|
jerry_size_t size = sizeof (string_buf);
|
|
|
|
string_buf[0] = '\0';
|
|
string_buf[1] = '\0';
|
|
|
|
TEST_ASSERT (counter >= 1 && counter <= 3);
|
|
TEST_ASSERT (jerry_value_is_string (error_value));
|
|
TEST_ASSERT (jerry_get_string_size (error_value) == size);
|
|
TEST_ASSERT (jerry_string_to_char_buffer (error_value, string_buf, size) == size);
|
|
TEST_ASSERT (string_buf[0] == 'e' && string_buf[1] == (char) ('0' + counter));
|
|
break;
|
|
}
|
|
case 3:
|
|
{
|
|
TEST_ASSERT (counter == 1);
|
|
TEST_ASSERT (jerry_get_error_type (error_value) == JERRY_ERROR_RANGE);
|
|
break;
|
|
}
|
|
case 4:
|
|
{
|
|
TEST_ASSERT (mode == 4);
|
|
TEST_ASSERT (counter >= 1 && counter <= 2);
|
|
|
|
jerry_error_t error = (counter == 1) ? JERRY_ERROR_REFERENCE : JERRY_ERROR_TYPE;
|
|
TEST_ASSERT (jerry_get_error_type (error_value) == error);
|
|
break;
|
|
}
|
|
case 5:
|
|
case 6:
|
|
{
|
|
TEST_ASSERT (counter >= 1 && counter <= 2);
|
|
TEST_ASSERT (jerry_value_is_false (error_value));
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
TEST_ASSERT (mode == 8 || mode == 9);
|
|
TEST_ASSERT (counter == 1);
|
|
TEST_ASSERT (jerry_value_is_true (error_value));
|
|
break;
|
|
}
|
|
}
|
|
} /* vm_throw_callback */
|
|
|
|
static jerry_value_t
|
|
native_handler (const jerry_call_info_t *call_info_p, /**< call info */
|
|
const jerry_value_t args_p[], /**< arguments */
|
|
const jerry_length_t args_count) /**< arguments length */
|
|
{
|
|
(void) call_info_p;
|
|
(void) args_p;
|
|
TEST_ASSERT (args_count == 0);
|
|
|
|
if (mode == 7)
|
|
{
|
|
jerry_value_t result = jerry_create_error (JERRY_ERROR_COMMON, (const jerry_char_t *) "Error!");
|
|
|
|
TEST_ASSERT (!jerry_error_is_throw_captured (result));
|
|
jerry_error_set_throw_capture (result, false);
|
|
TEST_ASSERT (jerry_error_is_throw_captured (result));
|
|
return result;
|
|
}
|
|
|
|
jerry_char_t source[] = TEST_STRING_LITERAL ("throw false");
|
|
jerry_value_t result = jerry_eval (source, sizeof (source) - 1, JERRY_PARSE_NO_OPTS);
|
|
|
|
TEST_ASSERT (jerry_error_is_throw_captured (result));
|
|
|
|
if (mode == 6)
|
|
{
|
|
jerry_error_set_throw_capture (result, true);
|
|
TEST_ASSERT (!jerry_error_is_throw_captured (result));
|
|
}
|
|
return result;
|
|
} /* native_handler */
|
|
|
|
static void
|
|
do_eval (const char *script_p, /**< script to evaluate */
|
|
bool should_throw) /**< script throws an error */
|
|
{
|
|
jerry_value_t result = jerry_eval ((const jerry_char_t *) script_p, strlen (script_p), JERRY_PARSE_NO_OPTS);
|
|
TEST_ASSERT (jerry_value_is_error (result) == should_throw);
|
|
jerry_release_value (result);
|
|
} /* do_eval */
|
|
|
|
int
|
|
main (void)
|
|
{
|
|
TEST_INIT ();
|
|
|
|
/* Test stopping an infinite loop. */
|
|
if (!jerry_is_feature_enabled (JERRY_FEATURE_VM_THROW))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
jerry_init (JERRY_INIT_EMPTY);
|
|
|
|
jerry_set_vm_throw_callback (vm_throw_callback, (void *) &mode);
|
|
|
|
mode = 0;
|
|
counter = 0;
|
|
do_eval (TEST_STRING_LITERAL ("throw -5.6"), true);
|
|
TEST_ASSERT (counter == 1);
|
|
|
|
mode = 1;
|
|
counter = 0;
|
|
do_eval (TEST_STRING_LITERAL ("function f() { throw null }\n"
|
|
"function g() { f() }\n"
|
|
"g()\n"),
|
|
true);
|
|
TEST_ASSERT (counter == 1);
|
|
|
|
mode = 2;
|
|
counter = 0;
|
|
do_eval (TEST_STRING_LITERAL ("function f() { throw 'e1' }\n"
|
|
"function g() { try { f() } catch (e) { throw 'e2' } }\n"
|
|
"try { g() } catch (e) { throw 'e3' }\n"),
|
|
true);
|
|
TEST_ASSERT (counter == 3);
|
|
|
|
mode = 3;
|
|
counter = 0;
|
|
do_eval (TEST_STRING_LITERAL ("function f() { throw new RangeError() }\n"
|
|
"function g() { try { f() } finally { } }\n"
|
|
"try { g() } finally { }\n"),
|
|
true);
|
|
TEST_ASSERT (counter == 1);
|
|
|
|
mode = 4;
|
|
counter = 0;
|
|
do_eval (TEST_STRING_LITERAL ("function f() { unresolved }\n"
|
|
"function g() { try { f() } finally { null.member } }\n"
|
|
"try { g() } finally { }\n"),
|
|
true);
|
|
TEST_ASSERT (counter == 2);
|
|
|
|
/* Native functions may trigger the call twice: */
|
|
jerry_value_t global_object_value = jerry_get_global_object ();
|
|
jerry_value_t function_value = jerry_create_external_function (native_handler);
|
|
jerry_value_t function_name_value = jerry_create_string ((const jerry_char_t *) "native");
|
|
|
|
jerry_release_value (jerry_set_property (global_object_value, function_name_value, function_value));
|
|
jerry_release_value (function_name_value);
|
|
jerry_release_value (function_value);
|
|
jerry_release_value (global_object_value);
|
|
|
|
mode = 5;
|
|
counter = 0;
|
|
do_eval (TEST_STRING_LITERAL ("native()\n"), true);
|
|
TEST_ASSERT (counter == 1);
|
|
|
|
mode = 6;
|
|
counter = 0;
|
|
do_eval (TEST_STRING_LITERAL ("native()\n"), true);
|
|
TEST_ASSERT (counter == 2);
|
|
|
|
mode = 7;
|
|
counter = 0;
|
|
do_eval (TEST_STRING_LITERAL ("native()\n"), true);
|
|
TEST_ASSERT (counter == 0);
|
|
|
|
/* Built-in functions should not trigger the call twice: */
|
|
mode = 8;
|
|
counter = 0;
|
|
do_eval (TEST_STRING_LITERAL ("function f() { eval('eval(\\'throw true\\')') }\n"
|
|
"f()\n"),
|
|
true);
|
|
TEST_ASSERT (counter == 1);
|
|
|
|
mode = 9;
|
|
counter = 0;
|
|
do_eval (TEST_STRING_LITERAL ("function f() { [1].map(function() { throw true }) }\n"
|
|
"f()\n"),
|
|
true);
|
|
TEST_ASSERT (counter == 1);
|
|
|
|
jerry_value_t value = jerry_create_object ();
|
|
TEST_ASSERT (!jerry_error_is_throw_captured (value));
|
|
jerry_error_set_throw_capture (value, false);
|
|
TEST_ASSERT (!jerry_error_is_throw_captured (value));
|
|
jerry_release_value (value);
|
|
|
|
jerry_cleanup ();
|
|
return 0;
|
|
} /* main */
|