/* * Copyright 2016 WebAssembly Community Group participants * * 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 "common.h" #include #include #include #include #if COMPILER_IS_MSVC #include #include #endif #include "allocator.h" #define V(rtype, type1, type2, mem_size, code, NAME, text) \ [code] = {text, WASM_TYPE_##rtype, WASM_TYPE_##type1, WASM_TYPE_##type2, \ mem_size}, WasmOpcodeInfo g_wasm_opcode_info[] = {WASM_FOREACH_OPCODE(V)}; #undef V const char* g_wasm_kind_name[] = {"func", "table", "memory", "global"}; WASM_STATIC_ASSERT(WASM_ARRAY_SIZE(g_wasm_kind_name) == WASM_NUM_EXTERNAL_KINDS); WasmBool wasm_is_naturally_aligned(WasmOpcode opcode, uint32_t alignment) { uint32_t opcode_align = wasm_get_opcode_memory_size(opcode); return alignment == WASM_USE_NATURAL_ALIGNMENT || alignment == opcode_align; } uint32_t wasm_get_opcode_alignment(WasmOpcode opcode, uint32_t alignment) { if (alignment == WASM_USE_NATURAL_ALIGNMENT) return wasm_get_opcode_memory_size(opcode); return alignment; } WasmStringSlice wasm_empty_string_slice(void) { WasmStringSlice result; result.start = ""; result.length = 0; return result; } WasmBool wasm_string_slice_eq_cstr(const WasmStringSlice* s1, const char* s2) { size_t s2_len = strlen(s2); if (s2_len != s1->length) return WASM_FALSE; return strncmp(s1->start, s2, s2_len) == 0 ? WASM_TRUE : WASM_FALSE; } WasmBool wasm_string_slice_startswith(const WasmStringSlice* s1, const char* s2) { size_t s2_len = strlen(s2); if (s2_len > s1->length) return WASM_FALSE; return strncmp(s1->start, s2, s2_len) == 0 ? WASM_TRUE : WASM_FALSE; } WasmStringSlice wasm_string_slice_from_cstr(const char* string) { WasmStringSlice result; result.start = string; result.length = strlen(string); return result; } WasmBool wasm_string_slice_is_empty(const WasmStringSlice* str) { assert(str); return str->start == NULL || str->length == 0; } WasmBool wasm_string_slices_are_equal(const WasmStringSlice* a, const WasmStringSlice* b) { assert(a && b); return a->start && b->start && a->length == b->length && memcmp(a->start, b->start, a->length) == 0; } void wasm_destroy_string_slice(WasmAllocator* allocator, WasmStringSlice* str) { assert(str); wasm_free(allocator, (void*)str->start); } WasmResult wasm_read_file(WasmAllocator* allocator, const char* filename, void** out_data, size_t* out_size) { FILE* infile = fopen(filename, "rb"); if (!infile) { fprintf(stderr, "unable to read file: %s\n", filename); return WASM_ERROR; } if (fseek(infile, 0, SEEK_END) < 0) { fprintf(stderr, "fseek to end failed.\n"); return WASM_ERROR; } long size = ftell(infile); if (size < 0) { fprintf(stderr, "ftell failed.\n"); return WASM_ERROR; } if (fseek(infile, 0, SEEK_SET) < 0) { fprintf(stderr, "fseek to beginning failed.\n"); return WASM_ERROR; } void* data = wasm_alloc(allocator, size, WASM_DEFAULT_ALIGN); if (size != 0 && fread(data, size, 1, infile) != 1) { fprintf(stderr, "fread failed.\n"); return WASM_ERROR; } *out_data = data; *out_size = size; fclose(infile); return WASM_OK; } static void print_carets(FILE* out, size_t num_spaces, size_t num_carets, size_t max_line) { /* print the caret */ char* carets = alloca(max_line); memset(carets, '^', max_line); if (num_carets > max_line - num_spaces) num_carets = max_line - num_spaces; /* always print at least one caret */ if (num_carets == 0) num_carets = 1; fprintf(out, "%*s%.*s\n", (int)num_spaces, "", (int)num_carets, carets); } static void print_source_error(FILE* out, const WasmLocation* loc, const char* error, const char* source_line, size_t source_line_length, size_t source_line_column_offset) { fprintf(out, "%s:%d:%d: %s\n", loc->filename, loc->line, loc->first_column, error); if (source_line && source_line_length > 0) { fprintf(out, "%s\n", source_line); size_t num_spaces = (loc->first_column - 1) - source_line_column_offset; size_t num_carets = loc->last_column - loc->first_column; print_carets(out, num_spaces, num_carets, source_line_length); } } static void print_error_header(FILE* out, WasmDefaultErrorHandlerInfo* info) { if (info && info->header) { switch (info->print_header) { case WASM_PRINT_ERROR_HEADER_NEVER: break; case WASM_PRINT_ERROR_HEADER_ONCE: info->print_header = WASM_PRINT_ERROR_HEADER_NEVER; /* Fallthrough. */ case WASM_PRINT_ERROR_HEADER_ALWAYS: fprintf(out, "%s:\n", info->header); break; } /* If there's a header, indent the following message. */ fprintf(out, " "); } } static FILE* get_default_error_handler_info_output_file( WasmDefaultErrorHandlerInfo* info) { return info && info->out_file ? info->out_file : stderr; } void wasm_default_source_error_callback(const WasmLocation* loc, const char* error, const char* source_line, size_t source_line_length, size_t source_line_column_offset, void* user_data) { WasmDefaultErrorHandlerInfo* info = user_data; FILE* out = get_default_error_handler_info_output_file(info); print_error_header(out, info); print_source_error(out, loc, error, source_line, source_line_length, source_line_column_offset); } void wasm_default_binary_error_callback(uint32_t offset, const char* error, void* user_data) { WasmDefaultErrorHandlerInfo* info = user_data; FILE* out = get_default_error_handler_info_output_file(info); print_error_header(out, info); if (offset == WASM_UNKNOWN_OFFSET) fprintf(out, "error: %s\n", error); else fprintf(out, "error: @0x%08x: %s\n", offset, error); fflush(out); } void wasm_init_stdio() { #if COMPILER_IS_MSVC int result = _setmode(_fileno(stdout), _O_BINARY); if (result == -1) perror("Cannot set mode binary to stdout"); result = _setmode(_fileno(stderr), _O_BINARY); if (result == -1) perror("Cannot set mode binary to stderr"); #endif }