mirror of
https://github.com/c3lang/c3c.git
synced 2026-02-27 12:01:16 +00:00
287 lines
6.5 KiB
C
287 lines
6.5 KiB
C
//
|
|
// Created by Christoffer Lerno on 2019-08-24.
|
|
//
|
|
|
|
#include "compiler_internal.h"
|
|
|
|
|
|
CompilationUnit *unit_create(File *file)
|
|
{
|
|
CompilationUnit *unit = CALLOCS(CompilationUnit);
|
|
unit->file = file;
|
|
unit->is_interface_file = str_has_suffix(file->name, ".c3i");
|
|
htable_init(&unit->local_symbols, 1024);
|
|
return unit;
|
|
}
|
|
|
|
|
|
static inline bool create_module_or_check_name(CompilationUnit *unit, Path *module_name, const char **parameters)
|
|
{
|
|
Module *module = unit->module;
|
|
if (!module)
|
|
{
|
|
module = unit->module = compiler_find_or_create_module(module_name, parameters);
|
|
}
|
|
else
|
|
{
|
|
if (unit->module->name->module != module_name->module)
|
|
{
|
|
SEMA_ERROR(module_name,
|
|
"Module name here '%s' did not match actual module '%s'.",
|
|
module_name->module,
|
|
module->name->module);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
vec_add(module->units, unit);
|
|
return true;
|
|
}
|
|
|
|
static bool filename_to_module_in_buffer(const char *path)
|
|
{
|
|
int len = (int)strlen(path);
|
|
int last_slash = 0;
|
|
int last_dot = -1;
|
|
for (int i = 0; i < len; i++)
|
|
{
|
|
if (path[i] == '/') last_slash = i;
|
|
if (path[i] == '.') last_dot = i;
|
|
}
|
|
int name_len = last_dot - last_slash - 1;
|
|
if (name_len < 1) return false;
|
|
scratch_buffer_clear();
|
|
for (int i = last_slash + 1; i < last_dot; i++)
|
|
{
|
|
char c = path[i];
|
|
if (char_is_letter(c) || char_is_digit(c))
|
|
{
|
|
c = (char)(char_is_upper(c) ? c + 'a' - 'A' : c);
|
|
}
|
|
else
|
|
{
|
|
c = '_';
|
|
}
|
|
scratch_buffer_append_char(c);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool context_set_module_from_filename(ParseContext *context)
|
|
{
|
|
File *file = context->unit->file;
|
|
if (!filename_to_module_in_buffer(file->full_path))
|
|
{
|
|
sema_error(context, "The filename '%s' could not be converted to a valid module name, try using an explicit module name.", file->full_path);
|
|
return false;
|
|
}
|
|
|
|
TokenType type = TOKEN_IDENT;
|
|
const char *module_name = symtab_add(scratch_buffer.str,
|
|
scratch_buffer.len,
|
|
fnv1a(scratch_buffer.str, (uint32_t) scratch_buffer.len),
|
|
&type);
|
|
|
|
if (type != TOKEN_IDENT)
|
|
{
|
|
sema_error(context, "Generating a filename from the file '%s' resulted in a name that is a reserved keyword, "
|
|
"try using an explicit module name.", file->full_path);
|
|
return false;
|
|
}
|
|
Path *path = CALLOCS(Path);
|
|
path->span = INVALID_SPAN;
|
|
path->module = module_name;
|
|
path->len = scratch_buffer.len;
|
|
return create_module_or_check_name(context->unit, path, NULL);
|
|
}
|
|
|
|
bool context_set_module(ParseContext *context, Path *path, const char **generic_parameters)
|
|
{
|
|
// Note that we allow the illegal name for now, to be able to parse further.
|
|
if (!str_has_no_uppercase(path->module))
|
|
{
|
|
SEMA_ERROR(path, "A module name may not have any uppercase characters.");
|
|
return false;
|
|
}
|
|
|
|
return create_module_or_check_name(context->unit, path, generic_parameters);
|
|
}
|
|
|
|
|
|
void unit_register_external_symbol(CompilationUnit *unit, Decl *decl)
|
|
{
|
|
if (!decl->unit || decl->unit->module == unit->module || !decl->extname) return;
|
|
decl->is_external_visible = true;
|
|
}
|
|
|
|
|
|
void decl_register(Decl *decl)
|
|
{
|
|
if (decl->visibility > VISIBLE_PUBLIC) return;
|
|
switch (decl->decl_kind)
|
|
{
|
|
case DECL_ERASED:
|
|
return;
|
|
case DECL_POISONED:
|
|
case DECL_CT_ASSERT:
|
|
case DECL_CT_ECHO:
|
|
case DECL_CT_EXEC:
|
|
case DECL_ENUM_CONSTANT:
|
|
case DECL_FAULTVALUE:
|
|
case DECL_IMPORT:
|
|
case DECL_LABEL:
|
|
case DECL_DECLARRAY:
|
|
case DECL_BODYPARAM:
|
|
case DECL_CT_INCLUDE:
|
|
case DECL_GLOBALS:
|
|
UNREACHABLE
|
|
case DECL_ATTRIBUTE:
|
|
case DECL_BITSTRUCT:
|
|
case DECL_DISTINCT:
|
|
case DECL_FAULT:
|
|
case DECL_ENUM:
|
|
case DECL_STRUCT:
|
|
case DECL_TYPEDEF:
|
|
case DECL_UNION:
|
|
case DECL_DEFINE:
|
|
case DECL_FUNC:
|
|
case DECL_MACRO:
|
|
case DECL_VAR:
|
|
case DECL_FNTYPE:
|
|
case DECL_PROTOCOL:
|
|
global_context_add_decl(decl);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void unit_register_global_decl(CompilationUnit *unit, Decl *decl)
|
|
{
|
|
assert(!decl->unit || decl->unit->module->is_generic);
|
|
decl->unit = unit;
|
|
|
|
switch (decl->decl_kind)
|
|
{
|
|
case DECL_ERASED:
|
|
return;
|
|
case DECL_POISONED:
|
|
break;
|
|
case DECL_MACRO:
|
|
assert(decl->name);
|
|
if (decl->func_decl.type_parent)
|
|
{
|
|
vec_add(unit->macro_methods, decl);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
vec_add(unit->macros, decl);
|
|
}
|
|
decl_set_external_name(decl);
|
|
decl_register(decl);
|
|
break;
|
|
case DECL_FUNC:
|
|
assert(decl->name);
|
|
if (decl->func_decl.type_parent)
|
|
{
|
|
vec_add(unit->methods, decl);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
vec_add(unit->functions, decl);
|
|
}
|
|
decl_register(decl);
|
|
break;
|
|
case DECL_VAR:
|
|
assert(decl->name);
|
|
vec_add(unit->vars, decl);
|
|
decl_set_external_name(decl);
|
|
decl_register(decl);
|
|
break;
|
|
case DECL_PROTOCOL:
|
|
case DECL_DISTINCT:
|
|
case DECL_STRUCT:
|
|
case DECL_UNION:
|
|
case DECL_TYPEDEF:
|
|
case DECL_FAULT:
|
|
case DECL_BITSTRUCT:
|
|
assert(decl->name);
|
|
vec_add(unit->types, decl);
|
|
decl_set_external_name(decl);
|
|
decl_register(decl);
|
|
break;
|
|
case DECL_DEFINE:
|
|
assert(decl->name);
|
|
vec_add(unit->generic_defines, decl);
|
|
decl_set_external_name(decl);
|
|
decl_register(decl);
|
|
break;
|
|
case DECL_ENUM:
|
|
assert(decl->name);
|
|
vec_add(unit->enums, decl);
|
|
decl_set_external_name(decl);
|
|
decl_register(decl);
|
|
break;
|
|
case DECL_ATTRIBUTE:
|
|
vec_add(unit->attributes, decl);
|
|
decl_register(decl);
|
|
break;
|
|
|
|
case DECL_FAULTVALUE:
|
|
case DECL_ENUM_CONSTANT:
|
|
case DECL_IMPORT:
|
|
case DECL_LABEL:
|
|
case DECL_DECLARRAY:
|
|
case DECL_BODYPARAM:
|
|
case DECL_GLOBALS:
|
|
case DECL_FNTYPE:
|
|
UNREACHABLE
|
|
case DECL_CT_EXEC:
|
|
case DECL_CT_INCLUDE:
|
|
vec_add(unit->ct_includes, decl);
|
|
return;
|
|
case DECL_CT_ECHO:
|
|
vec_add(unit->ct_echos, decl);
|
|
return;
|
|
case DECL_CT_ASSERT:
|
|
vec_add(unit->ct_asserts, decl);
|
|
return;
|
|
}
|
|
DEBUG_LOG("Registering symbol '%s' in %s.", decl->name, unit->module->name->module);
|
|
|
|
Decl *old;
|
|
if ((old = htable_set(&unit->local_symbols, (void*)decl->name, decl))) goto ERR;
|
|
if (decl->visibility < VISIBLE_LOCAL)
|
|
{
|
|
if ((old = htable_set(&unit->module->symbols, (void*)decl->name, decl))) goto ERR;
|
|
}
|
|
return;
|
|
ERR:
|
|
assert(decl != old);
|
|
sema_shadow_error(decl, old);
|
|
decl_poison(decl);
|
|
decl_poison(old);
|
|
}
|
|
|
|
bool unit_add_import(CompilationUnit *unit, Path *path, bool private_import)
|
|
{
|
|
DEBUG_LOG("SEMA: Add import of '%s'.", path->module);
|
|
|
|
if (!str_has_no_uppercase(path->module))
|
|
{
|
|
SEMA_ERROR(path, "A module is not expected to have any uppercase characters, please change it.");
|
|
return false;
|
|
}
|
|
|
|
Decl *import = decl_calloc();
|
|
import->span = path->span;
|
|
import->decl_kind = DECL_IMPORT;
|
|
import->import.path = path;
|
|
import->import.import_private_as_public = private_import;
|
|
|
|
vec_add(unit->imports, import);
|
|
DEBUG_LOG("Added import %s", path->module);
|
|
return true;
|
|
}
|
|
|