/** * @file pm_newline_list.h * * A list of byte offsets of newlines in a string. * * When compiling the syntax tree, it's necessary to know the line and column * of many nodes. This is necessary to support things like error messages, * tracepoints, etc. * * It's possible that we could store the start line, start column, end line, and * end column on every node in addition to the offsets that we already store, * but that would be quite a lot of memory overhead. */ #ifndef PRISM_NEWLINE_LIST_H #define PRISM_NEWLINE_LIST_H #include "prism/defines.h" #include #include #include #include /** * A list of offsets of newlines in a string. The offsets are assumed to be * sorted/inserted in ascending order. */ typedef struct { /** A pointer to the start of the source string. */ const uint8_t *start; /** The number of offsets in the list. */ size_t size; /** The capacity of the list that has been allocated. */ size_t capacity; /** The list of offsets. */ size_t *offsets; } pm_newline_list_t; /** * A line and column in a string. */ typedef struct { /** The line number. */ size_t line; /** The column number. */ size_t column; } pm_line_column_t; /** * Initialize a new newline list with the given capacity. Returns true if the * allocation of the offsets succeeds, otherwise returns false. * * @param list The list to initialize. * @param start A pointer to the start of the source string. * @param capacity The initial capacity of the list. * @return True if the allocation of the offsets succeeds, otherwise false. */ bool pm_newline_list_init(pm_newline_list_t *list, const uint8_t *start, size_t capacity); /** * Append a new offset to the newline list. Returns true if the reallocation of * the offsets succeeds (if one was necessary), otherwise returns false. * * @param list The list to append to. * @param cursor A pointer to the offset to append. * @return True if the reallocation of the offsets succeeds (if one was * necessary), otherwise false. */ bool pm_newline_list_append(pm_newline_list_t *list, const uint8_t *cursor); /** * Conditionally append a new offset to the newline list, if the value passed in * is a newline. * * @param list The list to append to. * @param cursor A pointer to the offset to append. * @return True if the reallocation of the offsets succeeds (if one was * necessary), otherwise false. */ bool pm_newline_list_check_append(pm_newline_list_t *list, const uint8_t *cursor); /** * Returns the line and column of the given offset. If the offset is not in the * list, the line and column of the closest offset less than the given offset * are returned. * * @param list The list to search. * @param cursor A pointer to the offset to search for. * @return The line and column of the given offset. */ pm_line_column_t pm_newline_list_line_column(const pm_newline_list_t *list, const uint8_t *cursor); /** * Free the internal memory allocated for the newline list. * * @param list The list to free. */ void pm_newline_list_free(pm_newline_list_t *list); #endif