// for each lookahead symbol (fs), put action into tmap // returns 0 on success, -1 on conflict // ignores forall entries static int terminals_put(HStringMap *tmap, const HStringMap *fs, HLRAction *action) { int ret = 0; if (fs->epsilon_branch) { HLRAction *prev = tmap->epsilon_branch; if (prev && prev != action) { // conflict tmap->epsilon_branch = h_lr_conflict(tmap->arena, prev, action); ret = -1; } else { tmap->epsilon_branch = action; } } if (fs->end_branch) { HLRAction *prev = tmap->end_branch; if (prev && prev != action) { // conflict tmap->end_branch = h_lr_conflict(tmap->arena, prev, action); ret = -1; } else { tmap->end_branch = action; } } H_FOREACH(fs->char_branches, void *key, HStringMap *fs_) HStringMap *tmap_ = h_hashtable_get(tmap->char_branches, key); if (!tmap_) { tmap_ = h_stringmap_new(tmap->arena); h_hashtable_put(tmap->char_branches, key, tmap_); } if (terminals_put(tmap_, fs_, action) < 0) { ret = -1; } H_END_FOREACH return ret; }
HLRTable *h_lrtable_new(HAllocator *mm__, size_t nrows) { HArena *arena = h_new_arena(mm__, 0); // default blocksize assert(arena != NULL); HLRTable *ret = h_new(HLRTable, 1); ret->nrows = nrows; ret->ntmap = h_arena_malloc(arena, nrows * sizeof(HHashTable *)); ret->tmap = h_arena_malloc(arena, nrows * sizeof(HStringMap *)); ret->forall = h_arena_malloc(arena, nrows * sizeof(HLRAction *)); ret->inadeq = h_slist_new(arena); ret->arena = arena; ret->mm__ = mm__; for(size_t i=0; i<nrows; i++) { ret->ntmap[i] = h_hashtable_new(arena, h_eq_symbol, h_hash_symbol); ret->tmap[i] = h_stringmap_new(arena); ret->forall[i] = NULL; } return ret; }