mirror of
https://github.com/jerryscript-project/jerryscript.git
synced 2025-12-15 16:29:21 +00:00
Add C API to work with Array Buffers. The following methods are added: - jerry_value_is_arraybuffer - jerry_create_arraybuffer - jerry_arraybuffer_write - jerry_arraybuffer_read - jerry_get_arraybuffer_byte_length JerryScript-DCO-1.0-Signed-off-by: Peter Gal pgal.u-szeged@partner.samsung.com
264 lines
8.6 KiB
C
264 lines
8.6 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 "jerryscript-port.h"
|
|
#include "jerryscript-port-default.h"
|
|
#include "test-common.h"
|
|
|
|
/**
|
|
* Register a JavaScript value in the global object.
|
|
*/
|
|
static void
|
|
register_js_value (const char *name_p, /**< name of the function */
|
|
jerry_value_t value) /**< JS value */
|
|
{
|
|
jerry_value_t global_obj_val = jerry_get_global_object ();
|
|
|
|
jerry_value_t name_val = jerry_create_string ((const jerry_char_t *) name_p);
|
|
jerry_value_t result_val = jerry_set_property (global_obj_val, name_val, value);
|
|
TEST_ASSERT (jerry_value_is_boolean (result_val));
|
|
|
|
jerry_release_value (name_val);
|
|
jerry_release_value (global_obj_val);
|
|
|
|
jerry_release_value (result_val);
|
|
} /* register_js_value */
|
|
|
|
static jerry_value_t
|
|
assert_handler (const jerry_value_t func_obj_val, /**< function object */
|
|
const jerry_value_t this_val, /**< this arg */
|
|
const jerry_value_t args_p[], /**< function arguments */
|
|
const jerry_length_t args_cnt) /**< number of function arguments */
|
|
{
|
|
JERRY_UNUSED (func_obj_val);
|
|
JERRY_UNUSED (this_val);
|
|
|
|
if (args_cnt > 0
|
|
&& jerry_value_is_boolean (args_p[0])
|
|
&& jerry_get_boolean_value (args_p[0]))
|
|
{
|
|
return jerry_create_boolean (true);
|
|
}
|
|
|
|
if (args_cnt > 1
|
|
&& jerry_value_is_string (args_p[1]))
|
|
{
|
|
jerry_length_t utf8_sz = jerry_get_string_size (args_p[1]);
|
|
char string_from_utf8[utf8_sz];
|
|
string_from_utf8[utf8_sz] = 0;
|
|
|
|
jerry_string_to_char_buffer (args_p[1], (jerry_char_t *) string_from_utf8, utf8_sz);
|
|
|
|
printf ("JS assert: %s\n", string_from_utf8);
|
|
}
|
|
|
|
TEST_ASSERT (false);
|
|
} /* assert_handler */
|
|
|
|
/**
|
|
* Checks whether global object has arraybuffer.
|
|
*/
|
|
static bool
|
|
arraybuffer_is_available (void)
|
|
{
|
|
jerry_value_t global_obj_val = jerry_get_global_object ();
|
|
jerry_value_t prop_name = jerry_create_string ((const jerry_char_t *) "ArrayBuffer");
|
|
|
|
jerry_value_t prop_value = jerry_has_property (global_obj_val, prop_name);
|
|
bool has_prop = jerry_get_boolean_value (prop_value);
|
|
|
|
jerry_release_value (global_obj_val);
|
|
jerry_release_value (prop_name);
|
|
jerry_release_value (prop_value);
|
|
|
|
return has_prop;
|
|
} /* arraybuffer_is_available */
|
|
|
|
/**
|
|
* Test ArrayBuffer 'read' api call with various offset values.
|
|
*/
|
|
static void
|
|
test_read_with_offset (uint8_t offset) /**< offset for buffer read. */
|
|
{
|
|
const char *eval_arraybuffer_src_p = ("var array = new Uint8Array (15);"
|
|
"for (var i = 0; i < array.length; i++) { array[i] = i * 2; };"
|
|
"array.buffer");
|
|
jerry_value_t arraybuffer = jerry_eval ((jerry_char_t *) eval_arraybuffer_src_p,
|
|
strlen (eval_arraybuffer_src_p),
|
|
true);
|
|
|
|
TEST_ASSERT (!jerry_value_has_error_flag (arraybuffer));
|
|
TEST_ASSERT (jerry_value_is_arraybuffer (arraybuffer));
|
|
TEST_ASSERT (jerry_get_arraybuffer_byte_length (arraybuffer) == 15);
|
|
|
|
uint8_t buffer[20];
|
|
memset (buffer, 120, 20);
|
|
|
|
/* Try to copy more than the target buffer size. */
|
|
jerry_length_t copied = jerry_arraybuffer_read (arraybuffer, offset, buffer, 20);
|
|
TEST_ASSERT (copied == (jerry_length_t)(15 - offset));
|
|
|
|
for (uint8_t i = 0; i < copied; i++)
|
|
{
|
|
TEST_ASSERT (buffer[i] == (i + offset) * 2);
|
|
}
|
|
TEST_ASSERT (buffer[15 - offset] == 120);
|
|
|
|
jerry_release_value (arraybuffer);
|
|
} /* test_read_with_offset */
|
|
|
|
/**
|
|
* Test ArrayBuffer 'write' api call with various offset values.
|
|
*/
|
|
static void test_write_with_offset (uint8_t offset) /**< offset for buffer write. */
|
|
{
|
|
{
|
|
jerry_value_t offset_val = jerry_create_number (offset);
|
|
register_js_value ("offset", offset_val);
|
|
jerry_release_value (offset_val);
|
|
|
|
}
|
|
|
|
const char *eval_arraybuffer_src_p = "var array = new Uint8Array (15); array.buffer";
|
|
jerry_value_t arraybuffer = jerry_eval ((jerry_char_t *) eval_arraybuffer_src_p,
|
|
strlen (eval_arraybuffer_src_p),
|
|
true);
|
|
|
|
TEST_ASSERT (!jerry_value_has_error_flag (arraybuffer));
|
|
TEST_ASSERT (jerry_value_is_arraybuffer (arraybuffer));
|
|
TEST_ASSERT (jerry_get_arraybuffer_byte_length (arraybuffer) == 15);
|
|
|
|
uint8_t buffer[20];
|
|
|
|
for (uint8_t i = 0; i < 20; i++)
|
|
{
|
|
buffer[i] = (uint8_t)(i * 3);
|
|
}
|
|
|
|
/* Intentionally copy more than the allowed space. */
|
|
jerry_length_t copied = jerry_arraybuffer_write (arraybuffer, offset, buffer, 20);
|
|
TEST_ASSERT (copied == (jerry_length_t)(15 - offset));
|
|
|
|
const char *eval_test_arraybuffer_p = (
|
|
"for (var i = 0; i < offset; i++)"
|
|
"{"
|
|
" assert (array[i] == 0, 'offset check for: ' + i + ' was: ' + array[i] + ' should be: 0');"
|
|
"};"
|
|
"for (var i = offset; i < array.length; i++)"
|
|
"{"
|
|
" var expected = (i - offset) * 3;"
|
|
" assert (array[i] == expected, 'calc check for: ' + i + ' was: ' + array[i] + ' should be: ' + expected);"
|
|
"};"
|
|
"assert (array[15] === undefined, 'ArrayBuffer out of bounds index should return undefined value');");
|
|
jerry_value_t res = jerry_eval ((jerry_char_t *) eval_test_arraybuffer_p,
|
|
strlen (eval_test_arraybuffer_p),
|
|
true);
|
|
jerry_release_value (res);
|
|
jerry_release_value (arraybuffer);
|
|
} /* test_write_with_offset */
|
|
|
|
int
|
|
main (void)
|
|
{
|
|
jerry_init (JERRY_INIT_EMPTY);
|
|
|
|
if (!arraybuffer_is_available ())
|
|
{
|
|
jerry_port_log (JERRY_LOG_LEVEL_ERROR, "ArrayBuffer is disabled!\n");
|
|
jerry_cleanup ();
|
|
return 0;
|
|
}
|
|
|
|
jerry_value_t function_val = jerry_create_external_function (assert_handler);
|
|
register_js_value ("assert", function_val);
|
|
jerry_release_value (function_val);
|
|
|
|
/* Test array buffer queries */
|
|
{
|
|
const char *eval_arraybuffer_src_p = "new ArrayBuffer (10)";
|
|
jerry_value_t eval_arraybuffer = jerry_eval ((jerry_char_t *) eval_arraybuffer_src_p,
|
|
strlen (eval_arraybuffer_src_p),
|
|
true);
|
|
TEST_ASSERT (!jerry_value_has_error_flag (eval_arraybuffer));
|
|
TEST_ASSERT (jerry_value_is_arraybuffer (eval_arraybuffer));
|
|
TEST_ASSERT (jerry_get_arraybuffer_byte_length (eval_arraybuffer) == 10);
|
|
jerry_release_value (eval_arraybuffer);
|
|
}
|
|
|
|
/* Test array buffer creation */
|
|
{
|
|
const uint32_t length = 15;
|
|
jerry_value_t arraybuffer = jerry_create_arraybuffer (length);
|
|
TEST_ASSERT (!jerry_value_has_error_flag (arraybuffer));
|
|
TEST_ASSERT (jerry_value_is_arraybuffer (arraybuffer));
|
|
TEST_ASSERT (jerry_get_arraybuffer_byte_length (arraybuffer) == length);
|
|
jerry_release_value (arraybuffer);
|
|
}
|
|
|
|
/* Test array buffer read operations */
|
|
for (uint8_t i = 0; i < 15; i++)
|
|
{
|
|
test_read_with_offset (i);
|
|
}
|
|
|
|
/* Test zero length ArrayBuffer read */
|
|
{
|
|
const uint32_t length = 0;
|
|
jerry_value_t arraybuffer = jerry_create_arraybuffer (length);
|
|
TEST_ASSERT (!jerry_value_has_error_flag (arraybuffer));
|
|
TEST_ASSERT (jerry_value_is_arraybuffer (arraybuffer));
|
|
TEST_ASSERT (jerry_get_arraybuffer_byte_length (arraybuffer) == length);
|
|
|
|
uint8_t data[20];
|
|
memset (data, 11, 20);
|
|
|
|
jerry_length_t bytes_read = jerry_arraybuffer_read (arraybuffer, 0, data, 20);
|
|
TEST_ASSERT (bytes_read == 0);
|
|
|
|
for (int i = 0; i < 20; i++)
|
|
{
|
|
TEST_ASSERT (data[i] == 11);
|
|
}
|
|
|
|
jerry_release_value (arraybuffer);
|
|
}
|
|
|
|
/* Test array buffer write operations */
|
|
for (uint8_t i = 0; i < 15; i++)
|
|
{
|
|
test_write_with_offset (i);
|
|
}
|
|
|
|
/* Test zero length ArrayBuffer write */
|
|
{
|
|
const uint32_t length = 0;
|
|
jerry_value_t arraybuffer = jerry_create_arraybuffer (length);
|
|
TEST_ASSERT (!jerry_value_has_error_flag (arraybuffer));
|
|
TEST_ASSERT (jerry_value_is_arraybuffer (arraybuffer));
|
|
TEST_ASSERT (jerry_get_arraybuffer_byte_length (arraybuffer) == length);
|
|
|
|
uint8_t data[20];
|
|
memset (data, 11, 20);
|
|
|
|
jerry_length_t bytes_written = jerry_arraybuffer_write (arraybuffer, 0, data, 20);
|
|
TEST_ASSERT (bytes_written == 0);
|
|
|
|
jerry_release_value (arraybuffer);
|
|
}
|
|
|
|
jerry_cleanup ();
|
|
} /* main */
|