2023-08-29 00:03:08 -04:00
|
|
|
#ifdef GEN_INTELLISENSE_DIRECTIVES
|
|
|
|
# pragma once
|
|
|
|
# include "gen.hpp"
|
|
|
|
#endif
|
2023-08-21 21:40:23 -04:00
|
|
|
|
2023-08-21 20:27:00 -04:00
|
|
|
// This is a simple file reader that reads the entire file into memory.
|
|
|
|
// It has an extra option to skip the first few lines for undesired includes.
|
|
|
|
// This is done so that includes can be kept in dependency and component files so that intellisense works.
|
2024-10-25 02:59:56 -04:00
|
|
|
inline
|
2023-08-28 23:46:50 -04:00
|
|
|
Code scan_file( char const* path )
|
2023-07-24 22:19:21 -04:00
|
|
|
{
|
|
|
|
FileInfo file;
|
|
|
|
|
|
|
|
FileError error = file_open_mode( & file, EFileMode_READ, path );
|
|
|
|
if ( error != EFileError_NONE )
|
|
|
|
{
|
2023-08-09 18:47:59 -04:00
|
|
|
GEN_FATAL( "scan_file: Could not open: %s", path );
|
2023-07-24 22:19:21 -04:00
|
|
|
}
|
|
|
|
|
2024-10-27 18:58:37 -04:00
|
|
|
ssize fsize = file_size( & file );
|
2023-07-24 22:19:21 -04:00
|
|
|
if ( fsize <= 0 )
|
|
|
|
{
|
2023-08-09 18:47:59 -04:00
|
|
|
GEN_FATAL("scan_file: %s is empty", path );
|
2023-07-24 22:19:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
String str = String::make_reserve( GlobalAllocator, fsize );
|
|
|
|
file_read( & file, str, fsize );
|
|
|
|
str.get_header().Length = fsize;
|
|
|
|
|
2023-08-28 23:46:50 -04:00
|
|
|
// Skip GEN_INTELLISENSE_DIRECTIVES preprocessor blocks
|
|
|
|
// Its designed so that the directive should be the first thing in the file.
|
|
|
|
// Anything that comes before it will also be omitted.
|
2023-08-21 20:27:00 -04:00
|
|
|
{
|
|
|
|
#define current (*scanner)
|
2023-08-28 23:46:50 -04:00
|
|
|
#define matched 0
|
|
|
|
#define move_fwd() do { ++ scanner; -- left; } while (0)
|
|
|
|
const StrC directive_start = txt( "ifdef" );
|
|
|
|
const StrC directive_end = txt( "endif" );
|
|
|
|
const StrC def_intellisense = txt("GEN_INTELLISENSE_DIRECTIVES" );
|
|
|
|
|
|
|
|
bool found_directive = false;
|
|
|
|
char const* scanner = str.Data;
|
|
|
|
s32 left = fsize;
|
|
|
|
while ( left )
|
2023-08-21 20:27:00 -04:00
|
|
|
{
|
2023-08-28 23:46:50 -04:00
|
|
|
// Processing directive.
|
|
|
|
if ( current == '#' )
|
2023-08-21 20:27:00 -04:00
|
|
|
{
|
2023-08-28 23:46:50 -04:00
|
|
|
move_fwd();
|
|
|
|
while ( left && char_is_space( current ) )
|
|
|
|
move_fwd();
|
2023-08-21 20:27:00 -04:00
|
|
|
|
2023-08-28 23:46:50 -04:00
|
|
|
if ( ! found_directive )
|
2023-08-21 20:27:00 -04:00
|
|
|
{
|
2023-08-28 23:46:50 -04:00
|
|
|
if ( left && str_compare( scanner, directive_start.Ptr, directive_start.Len ) == matched )
|
2023-08-21 20:27:00 -04:00
|
|
|
{
|
2023-08-28 23:46:50 -04:00
|
|
|
scanner += directive_start.Len;
|
|
|
|
left -= directive_start.Len;
|
2023-08-21 20:27:00 -04:00
|
|
|
|
2023-08-28 23:46:50 -04:00
|
|
|
while ( left && char_is_space( current ) )
|
|
|
|
move_fwd();
|
2023-08-21 22:48:05 -04:00
|
|
|
|
2023-08-28 23:46:50 -04:00
|
|
|
if ( left && str_compare( scanner, def_intellisense.Ptr, def_intellisense.Len ) == matched )
|
|
|
|
{
|
|
|
|
scanner += def_intellisense.Len;
|
|
|
|
left -= def_intellisense.Len;
|
|
|
|
|
|
|
|
found_directive = true;
|
|
|
|
}
|
2023-08-21 20:27:00 -04:00
|
|
|
}
|
2023-08-28 23:46:50 -04:00
|
|
|
|
|
|
|
// Skip to end of line
|
|
|
|
while ( left && current != '\r' && current != '\n' )
|
|
|
|
move_fwd();
|
|
|
|
move_fwd();
|
|
|
|
|
|
|
|
if ( left && current == '\n' )
|
|
|
|
move_fwd();
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( left && str_compare( scanner, directive_end.Ptr, directive_end.Len ) == matched )
|
|
|
|
{
|
|
|
|
scanner += directive_end.Len;
|
|
|
|
left -= directive_end.Len;
|
|
|
|
|
|
|
|
// Skip to end of line
|
|
|
|
while ( left && current != '\r' && current != '\n' )
|
|
|
|
move_fwd();
|
|
|
|
move_fwd();
|
|
|
|
|
|
|
|
if ( left && current == '\n' )
|
|
|
|
move_fwd();
|
|
|
|
|
2023-09-07 22:51:15 -04:00
|
|
|
// sptr skip_size = fsize - left;
|
2023-08-28 23:46:50 -04:00
|
|
|
if ( (scanner + 2) >= ( str.Data + fsize ) )
|
2023-08-21 20:27:00 -04:00
|
|
|
{
|
2023-08-28 23:46:50 -04:00
|
|
|
mem_move( str, scanner, left );
|
|
|
|
str.get_header().Length = left;
|
|
|
|
break;
|
2023-08-21 20:27:00 -04:00
|
|
|
}
|
|
|
|
|
2023-08-28 23:46:50 -04:00
|
|
|
mem_move( str, scanner, left );
|
|
|
|
str.get_header().Length = left;
|
2023-08-21 20:27:00 -04:00
|
|
|
|
2023-08-28 23:46:50 -04:00
|
|
|
break;
|
2023-08-21 20:27:00 -04:00
|
|
|
}
|
2023-08-28 23:46:50 -04:00
|
|
|
|
2023-08-21 20:27:00 -04:00
|
|
|
}
|
|
|
|
|
2023-08-28 23:46:50 -04:00
|
|
|
move_fwd();
|
2023-08-21 20:27:00 -04:00
|
|
|
}
|
2023-08-28 23:46:50 -04:00
|
|
|
#undef move_fwd
|
|
|
|
#undef matched
|
2023-08-21 20:27:00 -04:00
|
|
|
#undef current
|
|
|
|
}
|
2023-07-24 22:19:21 -04:00
|
|
|
|
2023-08-21 20:27:00 -04:00
|
|
|
file_close( & file );
|
2023-07-24 22:19:21 -04:00
|
|
|
return untyped_str( str );
|
|
|
|
}
|
|
|
|
|
2023-08-03 23:18:33 -04:00
|
|
|
#if 0
|
2023-09-11 23:22:53 -04:00
|
|
|
struct CodeFile
|
2023-07-24 22:19:21 -04:00
|
|
|
{
|
2023-09-11 23:22:53 -04:00
|
|
|
using namespace Parser;
|
2023-07-18 23:33:00 -04:00
|
|
|
|
2023-09-11 23:22:53 -04:00
|
|
|
String FilePath;
|
|
|
|
TokArray Tokens;
|
|
|
|
Array<ParseFailure> ParseFailures;
|
|
|
|
Code CodeRoot;
|
2023-07-18 23:33:00 -04:00
|
|
|
};
|
|
|
|
|
2023-09-11 23:22:53 -04:00
|
|
|
namespace Parser
|
2023-07-18 23:33:00 -04:00
|
|
|
{
|
2023-09-11 23:22:53 -04:00
|
|
|
struct ParseFailure
|
2023-07-18 23:33:00 -04:00
|
|
|
{
|
2023-09-11 23:22:53 -04:00
|
|
|
String Reason;
|
|
|
|
Code Node;
|
2023-07-18 23:33:00 -04:00
|
|
|
};
|
2023-09-11 23:22:53 -04:00
|
|
|
}
|
2023-07-18 23:33:00 -04:00
|
|
|
|
2023-09-11 23:22:53 -04:00
|
|
|
CodeFile scan_file( char const* path )
|
|
|
|
{
|
|
|
|
using namespace Parser;
|
2023-07-18 23:33:00 -04:00
|
|
|
|
2023-09-11 23:22:53 -04:00
|
|
|
CodeFile
|
|
|
|
result = {};
|
|
|
|
result.FilePath = String::make( GlobalAllocator, path );
|
2023-07-18 23:33:00 -04:00
|
|
|
|
2023-09-11 23:22:53 -04:00
|
|
|
Code code = scan_file( path );
|
|
|
|
result.CodeRoot = code;
|
2023-07-18 23:33:00 -04:00
|
|
|
|
2023-09-11 23:22:53 -04:00
|
|
|
ParseContext context = parser_get_last_context();
|
|
|
|
result.Tokens = context.Tokens;
|
|
|
|
result.ParseFailures = context.Failures;
|
2023-07-18 23:33:00 -04:00
|
|
|
|
2023-09-11 23:22:53 -04:00
|
|
|
return result;
|
|
|
|
}
|
2023-08-03 23:18:33 -04:00
|
|
|
#endif
|