2020-02-15 23:38:06 +00:00
|
|
|
U0 LexBackupLastChar(CCompCtrl *cc)
|
2020-02-15 20:01:48 +00:00
|
|
|
{
|
2020-02-16 01:19:05 +00:00
|
|
|
CLexFile *tmpf=cc->lex_include_stack;
|
2020-02-15 20:01:48 +00:00
|
|
|
tmpf->buf_ptr=cc->cur_buf_ptr;
|
|
|
|
if (cc->flags & CCF_USE_LAST_U16) {
|
|
|
|
tmpf->last_U16=cc->last_U16;
|
|
|
|
cc->flags&=~CCF_USE_LAST_U16;
|
|
|
|
} else
|
|
|
|
tmpf->last_U16=0;
|
|
|
|
}
|
|
|
|
|
2020-02-15 23:38:06 +00:00
|
|
|
U0 LexPush(CCompCtrl *cc)
|
2020-02-15 20:01:48 +00:00
|
|
|
{//Create token-stream save point.
|
|
|
|
CLexFile *tmpf;
|
|
|
|
LexBackupLastChar(cc);
|
2020-02-16 01:19:05 +00:00
|
|
|
if (cc->lex_include_stack->last_U16)
|
2020-02-15 20:01:48 +00:00
|
|
|
cc->flags|=CCF_USE_LAST_U16;
|
2020-02-16 01:19:05 +00:00
|
|
|
tmpf=MAllocIdent(cc->lex_include_stack);
|
2020-02-19 02:10:39 +00:00
|
|
|
tmpf->next=cc->lex_parse_stack;
|
|
|
|
cc->lex_parse_stack=tmpf;
|
2020-02-15 20:01:48 +00:00
|
|
|
}
|
|
|
|
|
2020-02-15 23:38:06 +00:00
|
|
|
U0 LexPopRestore(CCompCtrl *cc)
|
2020-02-15 20:01:48 +00:00
|
|
|
{//Restore token-stream saved-point.
|
|
|
|
//Bad things can happen if you cross an #include file boundary.
|
2020-02-19 02:10:39 +00:00
|
|
|
CLexFile *tmpf=cc->lex_parse_stack;
|
2020-02-15 20:01:48 +00:00
|
|
|
cc->cur_buf_ptr=tmpf->buf_ptr;
|
|
|
|
if (cc->last_U16=tmpf->last_U16)
|
|
|
|
cc->flags|=CCF_USE_LAST_U16;
|
|
|
|
else
|
|
|
|
cc->flags&=~CCF_USE_LAST_U16;
|
2020-02-16 03:32:01 +00:00
|
|
|
MemCopy(cc->lex_include_stack(U8 *)+sizeof(U8 *),tmpf(U8 *)+sizeof(U8 *),
|
2020-02-15 20:01:48 +00:00
|
|
|
sizeof(CLexFile)-sizeof(U8 *));
|
2020-02-19 02:10:39 +00:00
|
|
|
cc->lex_parse_stack=tmpf->next;
|
2020-02-15 20:01:48 +00:00
|
|
|
Free(tmpf);
|
|
|
|
}
|
|
|
|
|
2020-02-15 23:38:06 +00:00
|
|
|
U0 LexPopNoRestore(CCompCtrl *cc)
|
2020-02-15 20:01:48 +00:00
|
|
|
{//Don't restore token-stream saved-point.
|
2020-02-19 02:10:39 +00:00
|
|
|
CLexFile *tmpf=cc->lex_parse_stack;
|
|
|
|
cc->lex_parse_stack=tmpf->next;
|
2020-02-15 20:01:48 +00:00
|
|
|
Free(tmpf);
|
|
|
|
}
|
|
|
|
|
2020-02-16 00:26:51 +00:00
|
|
|
I64 MemberMetaData(U8 *needle_str,CMemberList *haystack_member_list)
|
2020-02-16 04:57:03 +00:00
|
|
|
{//Find meta data name, return meta data val. See $LK,"::/Demo/ClassMeta.CC"$.
|
2020-02-16 00:26:51 +00:00
|
|
|
CMemberListMeta *meta=haystack_member_list->meta;
|
2020-02-15 20:01:48 +00:00
|
|
|
while (meta) {
|
2020-02-16 03:32:01 +00:00
|
|
|
if (!StrCompare(meta->str,needle_str))
|
2020-02-15 20:01:48 +00:00
|
|
|
return meta->user_data;
|
|
|
|
meta=meta->next;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-02-16 00:26:51 +00:00
|
|
|
CMemberListMeta *MemberMetaFind(U8 *needle_str,CMemberList *haystack_member_list)
|
2020-02-16 04:57:03 +00:00
|
|
|
{//Find meta data name, return meta data struct. See $LK,"::/Demo/ClassMeta.CC"$.
|
2020-02-16 00:26:51 +00:00
|
|
|
CMemberListMeta *meta=haystack_member_list->meta;
|
2020-02-15 20:01:48 +00:00
|
|
|
while (meta) {
|
2020-02-16 03:32:01 +00:00
|
|
|
if (!StrCompare(meta->str,needle_str))
|
2020-02-15 20:01:48 +00:00
|
|
|
return meta;
|
|
|
|
meta=meta->next;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-02-16 00:26:51 +00:00
|
|
|
CMemberList *MemberFind(U8 *needle_str,CHashClass *haystack_class)
|
2020-02-15 20:01:48 +00:00
|
|
|
{//Find class member. See $LK,"ClassRep",A="MN:ClassRep"$() and $LK,"DocForm",A="MN:DocForm"$().
|
|
|
|
I64 i;
|
2020-02-16 00:26:51 +00:00
|
|
|
CMemberList *tmpm;
|
2020-02-15 20:01:48 +00:00
|
|
|
do {
|
2020-02-16 00:26:51 +00:00
|
|
|
tmpm=haystack_class->member_list_and_root;
|
2020-02-15 20:01:48 +00:00
|
|
|
while (tmpm) {
|
2020-02-16 03:32:01 +00:00
|
|
|
if (!(i=StrCompare(tmpm->str,needle_str))) {
|
2020-02-16 00:20:04 +00:00
|
|
|
tmpm->use_count++;
|
2020-02-15 20:01:48 +00:00
|
|
|
return tmpm;
|
|
|
|
}
|
|
|
|
if (i<=0)
|
|
|
|
tmpm=tmpm->left;
|
|
|
|
else
|
|
|
|
tmpm=tmpm->right;
|
|
|
|
}
|
|
|
|
} while (haystack_class=haystack_class->base_class);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-02-16 00:26:51 +00:00
|
|
|
CMemberList *MemberClassBaseFind(CHashClass *needle_class,
|
2020-02-15 20:01:48 +00:00
|
|
|
CHashClass *haystack_class)
|
|
|
|
{//Find class member class base. For finding dup class local vars.
|
2020-02-16 00:26:51 +00:00
|
|
|
CMemberList *tmpm;
|
2020-02-15 20:01:48 +00:00
|
|
|
tmpm=haystack_class->member_class_base_root;
|
|
|
|
while (tmpm) {
|
|
|
|
if (needle_class==tmpm->member_class_base)
|
|
|
|
return tmpm;
|
|
|
|
if (needle_class<tmpm->member_class_base)
|
|
|
|
tmpm=tmpm->left_class_base;
|
|
|
|
else
|
|
|
|
tmpm=tmpm->right_class_base;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-02-16 00:26:51 +00:00
|
|
|
U0 MemberAdd(CCompCtrl *cc,CMemberList *tmpm,CHashClass *tmpc,I64 mode)
|
2020-02-15 20:01:48 +00:00
|
|
|
{
|
|
|
|
U8 *st=tmpm->str;
|
2020-02-16 00:26:51 +00:00
|
|
|
CMemberList **tmpm1,*tmpm2;
|
2020-02-15 20:01:48 +00:00
|
|
|
|
2020-02-16 03:32:01 +00:00
|
|
|
if (MemberFind(st,tmpc) && StrCompare(st,"pad") &&
|
|
|
|
StrCompare(st,"reserved") && StrCompare(st,"_anon_"))
|
2020-02-15 20:01:48 +00:00
|
|
|
LexExcept(cc,"Duplicate member at ");
|
2020-02-16 00:26:51 +00:00
|
|
|
tmpm1=&tmpc->member_list_and_root;
|
2020-02-15 20:01:48 +00:00
|
|
|
while (tmpm2=*tmpm1) {
|
2020-02-16 03:32:01 +00:00
|
|
|
if (StrCompare(tmpm2->str,st)<=0)
|
2020-02-15 20:01:48 +00:00
|
|
|
tmpm1=&tmpm2->left;
|
|
|
|
else
|
|
|
|
tmpm1=&tmpm2->right;
|
|
|
|
}
|
|
|
|
*tmpm1=tmpm;
|
|
|
|
|
|
|
|
if (mode==PRS1B_LOCAL_VAR) {
|
|
|
|
tmpm->member_class_base=
|
2020-02-16 00:20:04 +00:00
|
|
|
tmpm->member_class-tmpm->member_class->ptr_stars_count;
|
2020-02-15 20:01:48 +00:00
|
|
|
if (Bt(&cc->opts,OPTf_WARN_DUP_TYPES) &&
|
|
|
|
MemberClassBaseFind(tmpm->member_class_base,tmpc))
|
|
|
|
LexWarn(cc,"Duplicate type at ");
|
|
|
|
tmpm1=&tmpc->member_class_base_root;
|
|
|
|
while (tmpm2=*tmpm1) {
|
|
|
|
if (tmpm->member_class_base<tmpm2->member_class_base)
|
|
|
|
tmpm1=&tmpm2->left_class_base;
|
|
|
|
else if (tmpm->member_class_base>tmpm2->member_class_base)
|
|
|
|
tmpm1=&tmpm2->right_class_base;
|
|
|
|
else {
|
|
|
|
tmpm1=NULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (tmpm1)
|
|
|
|
*tmpm1=tmpm;
|
|
|
|
} else
|
|
|
|
tmpm->member_class_base=NULL;
|
|
|
|
|
|
|
|
tmpm->left=NULL;
|
|
|
|
tmpm->right=NULL;
|
|
|
|
tmpm->left_class_base=NULL;
|
|
|
|
tmpm->right_class_base=NULL;
|
2020-02-16 00:26:51 +00:00
|
|
|
tmpm2=tmpc->last_in_member_list;
|
|
|
|
tmpm2->next=tmpc->last_in_member_list=tmpm;
|
2020-02-15 20:01:48 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 00:26:51 +00:00
|
|
|
CMemberList *MemberListNew(I64 _reg)
|
2020-02-15 20:01:48 +00:00
|
|
|
{
|
2020-02-16 00:26:51 +00:00
|
|
|
CMemberList *res=CAlloc(sizeof(CMemberList));
|
2020-02-15 20:01:48 +00:00
|
|
|
res->reg=_reg;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2020-02-16 00:26:51 +00:00
|
|
|
Bool MemberListCmp(CMemberList *tmpm1,CMemberList *tmpm2,I64 count=I64_MAX)
|
2020-02-15 20:01:48 +00:00
|
|
|
{
|
2020-02-16 00:20:04 +00:00
|
|
|
while (tmpm1 && tmpm2 && count--) {
|
2020-02-16 03:32:01 +00:00
|
|
|
if (StrCompare(tmpm1->str,tmpm2->str) ||
|
2020-02-15 20:01:48 +00:00
|
|
|
tmpm1->member_class!=tmpm2->member_class ||
|
|
|
|
tmpm1->member_class_base!=tmpm2->member_class_base)
|
|
|
|
return FALSE;
|
2020-02-16 00:45:35 +00:00
|
|
|
if (tmpm1->flags&MLF_DEFAULT_AVAILABLE || tmpm2->flags&MLF_DEFAULT_AVAILABLE) {
|
|
|
|
if (tmpm1->flags&(MLF_DEFAULT_AVAILABLE|MLF_STR_DEFAULT_AVAILABLE)!=
|
|
|
|
tmpm2->flags&(MLF_DEFAULT_AVAILABLE|MLF_STR_DEFAULT_AVAILABLE))
|
2020-02-15 20:01:48 +00:00
|
|
|
return FALSE;
|
2020-02-16 00:45:35 +00:00
|
|
|
if (tmpm1->flags&MLF_STR_DEFAULT_AVAILABLE) {
|
2020-02-16 03:32:01 +00:00
|
|
|
if (StrCompare(tmpm1->default_val,tmpm2->default_val))
|
2020-02-15 20:01:48 +00:00
|
|
|
return FALSE;
|
2020-02-16 00:45:35 +00:00
|
|
|
} else if (tmpm1->default_val!=tmpm2->default_val)
|
2020-02-15 20:01:48 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
tmpm1=tmpm1->next;
|
|
|
|
tmpm2=tmpm2->next;
|
|
|
|
}
|
2020-02-16 00:20:04 +00:00
|
|
|
if (count<0 || !tmpm1 && !tmpm2)
|
2020-02-15 20:01:48 +00:00
|
|
|
return TRUE;
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2020-02-16 00:26:51 +00:00
|
|
|
U0 MemberListDel(CMemberList *tmpm)
|
2020-02-15 20:01:48 +00:00
|
|
|
{
|
2020-02-16 00:26:51 +00:00
|
|
|
CMemberList *tmpm1;
|
|
|
|
CMemberListMeta *tmp_meta,*tmp_meta1;
|
2020-02-15 20:01:48 +00:00
|
|
|
while (tmpm) {
|
|
|
|
tmpm1=tmpm->next;
|
|
|
|
Free(tmpm->str);
|
2020-02-16 00:26:51 +00:00
|
|
|
LinkedListDel(tmpm->dim.next);
|
2020-02-16 00:45:35 +00:00
|
|
|
if (tmpm->flags & MLF_STR_DEFAULT_AVAILABLE)
|
|
|
|
Free(tmpm->default_val);
|
2020-02-15 20:01:48 +00:00
|
|
|
if (tmpm->flags & MLF_FUN)
|
2020-02-16 00:20:04 +00:00
|
|
|
HashDel(tmpm->fun_ptr-tmpm->fun_ptr->ptr_stars_count);
|
2020-02-15 20:01:48 +00:00
|
|
|
tmp_meta=tmpm->meta;
|
|
|
|
while (tmp_meta) {
|
|
|
|
tmp_meta1=tmp_meta->next;
|
|
|
|
Free(tmp_meta->str);
|
|
|
|
if (tmp_meta->flags&MLMF_IS_STR)
|
|
|
|
Free(tmp_meta->user_data);
|
|
|
|
Free(tmp_meta);
|
|
|
|
tmp_meta=tmp_meta1;
|
|
|
|
}
|
|
|
|
Free(tmpm);
|
|
|
|
tmpm=tmpm1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-16 00:26:51 +00:00
|
|
|
U0 ClassMemberListDel(CHashClass *tmpc)
|
2020-02-15 20:01:48 +00:00
|
|
|
{
|
2020-02-16 00:26:51 +00:00
|
|
|
MemberListDel(tmpc->member_list_and_root);
|
2020-02-15 20:01:48 +00:00
|
|
|
tmpc->size=0;
|
2020-02-16 00:26:51 +00:00
|
|
|
tmpc->last_in_member_list=&tmpc->member_list_and_root;
|
|
|
|
tmpc->member_list_and_root=NULL;
|
2020-02-15 20:01:48 +00:00
|
|
|
tmpc->member_class_base_root=NULL;
|
2020-02-16 00:20:04 +00:00
|
|
|
tmpc->member_count=0;
|
2020-02-15 20:01:48 +00:00
|
|
|
if (tmpc->type&HTT_FUN)
|
2020-02-16 00:20:04 +00:00
|
|
|
tmpc(CHashFun *)->arg_count=0;
|
2020-02-15 20:01:48 +00:00
|
|
|
}
|
|
|
|
|
2020-02-16 00:26:51 +00:00
|
|
|
I64 MemberListSize(CHashClass *tmpc)
|
2020-02-15 20:01:48 +00:00
|
|
|
{
|
2020-02-16 00:26:51 +00:00
|
|
|
CMemberList *tmpm;
|
|
|
|
CMemberListMeta *tmp_meta;
|
2020-02-15 20:01:48 +00:00
|
|
|
I64 res=0;
|
2020-02-16 00:26:51 +00:00
|
|
|
tmpm=tmpc->member_list_and_root;
|
2020-02-15 20:01:48 +00:00
|
|
|
while (tmpm) {
|
|
|
|
res+=MSize2(tmpm->str);
|
2020-02-16 00:26:51 +00:00
|
|
|
res+=LinkedListSize(tmpm->dim.next);
|
2020-02-16 00:45:35 +00:00
|
|
|
if (tmpm->flags & MLF_STR_DEFAULT_AVAILABLE)
|
|
|
|
res+=MSize2(tmpm->default_val);
|
2020-02-15 20:01:48 +00:00
|
|
|
if (tmpm->flags & MLF_FUN)
|
2020-02-16 00:20:04 +00:00
|
|
|
res+=HashEntrySize2(tmpm->fun_ptr-tmpm->fun_ptr->ptr_stars_count);
|
2020-02-15 20:01:48 +00:00
|
|
|
tmp_meta=tmpm->meta;
|
|
|
|
while (tmp_meta) {
|
|
|
|
res+=MSize2(tmp_meta->str);
|
|
|
|
if (tmp_meta->flags&MLMF_IS_STR)
|
|
|
|
res+=MSize2(tmp_meta->user_data);
|
|
|
|
res+=MSize2(tmp_meta);
|
|
|
|
tmp_meta=tmp_meta->next;
|
|
|
|
}
|
|
|
|
res+=MSize2(tmpm);
|
|
|
|
tmpm=tmpm->next;
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2020-02-15 23:38:06 +00:00
|
|
|
U8 *LexExtStr(CCompCtrl *cc,I64 *_size=NULL,Bool lex_next=TRUE)
|
2020-02-15 20:01:48 +00:00
|
|
|
{//Lex $LK,"TK_STR",A="MN:TK_STR"$'s to one combined str. _size includes terminator.
|
|
|
|
I64 len=cc->cur_str_len,len1,len2;
|
|
|
|
U8 *st=cc->cur_str,*st1,*st2;
|
|
|
|
cc->cur_str=NULL;
|
|
|
|
while (cc->token==TK_STR) {
|
|
|
|
st1=st;
|
|
|
|
len1=len;
|
|
|
|
if (!lex_next && LexGetChar(cc)!='\\') {
|
|
|
|
cc->flags|=CCF_USE_LAST_U16;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (Lex(cc)==TK_STR) {
|
|
|
|
len2=cc->cur_str_len;
|
|
|
|
st2=cc->cur_str;
|
|
|
|
cc->cur_str=NULL;
|
|
|
|
len=len1+len2-1;
|
|
|
|
st=MAlloc(len);
|
|
|
|
if (len1>1)
|
2020-02-16 03:32:01 +00:00
|
|
|
MemCopy(st,st1,len1-1);
|
|
|
|
MemCopy(st+len1-1,st2,len2);
|
2020-02-15 20:01:48 +00:00
|
|
|
Free(st1);
|
|
|
|
Free(st2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (_size) *_size=len;
|
|
|
|
return st;
|
|
|
|
}
|