NCDAst.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874
  1. /**
  2. * @file NCDAst.c
  3. * @author Ambroz Bizjak <ambrop7@gmail.com>
  4. *
  5. * @section LICENSE
  6. *
  7. * Redistribution and use in source and binary forms, with or without
  8. * modification, are permitted provided that the following conditions are met:
  9. * 1. Redistributions of source code must retain the above copyright
  10. * notice, this list of conditions and the following disclaimer.
  11. * 2. Redistributions in binary form must reproduce the above copyright
  12. * notice, this list of conditions and the following disclaimer in the
  13. * documentation and/or other materials provided with the distribution.
  14. * 3. Neither the name of the author nor the
  15. * names of its contributors may be used to endorse or promote products
  16. * derived from this software without specific prior written permission.
  17. *
  18. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
  19. * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  20. * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  21. * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
  22. * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  23. * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  24. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  25. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  26. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  27. * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  28. */
  29. #include <stdlib.h>
  30. #include <limits.h>
  31. #include <string.h>
  32. #include <misc/offset.h>
  33. #include "NCDAst.h"
  34. struct NCDValue__list_element {
  35. LinkedList1Node list_node;
  36. NCDValue v;
  37. };
  38. struct NCDValue__map_element {
  39. LinkedList1Node list_node;
  40. NCDValue key;
  41. NCDValue val;
  42. };
  43. static void value_assert (NCDValue *o)
  44. {
  45. switch (o->type) {
  46. case NCDVALUE_STRING:
  47. case NCDVALUE_LIST:
  48. case NCDVALUE_MAP:
  49. case NCDVALUE_VAR:
  50. return;
  51. default:
  52. ASSERT(0);
  53. }
  54. }
  55. void NCDValue_Free (NCDValue *o)
  56. {
  57. switch (o->type) {
  58. case NCDVALUE_STRING: {
  59. free(o->string);
  60. } break;
  61. case NCDVALUE_LIST: {
  62. LinkedList1Node *n;
  63. while (n = LinkedList1_GetFirst(&o->list)) {
  64. struct NCDValue__list_element *e = UPPER_OBJECT(n, struct NCDValue__list_element, list_node);
  65. NCDValue_Free(&e->v);
  66. LinkedList1_Remove(&o->list, &e->list_node);
  67. free(e);
  68. }
  69. } break;
  70. case NCDVALUE_MAP: {
  71. LinkedList1Node *n;
  72. while (n = LinkedList1_GetFirst(&o->map_list)) {
  73. struct NCDValue__map_element *e = UPPER_OBJECT(n, struct NCDValue__map_element, list_node);
  74. LinkedList1_Remove(&o->map_list, &e->list_node);
  75. NCDValue_Free(&e->key);
  76. NCDValue_Free(&e->val);
  77. free(e);
  78. }
  79. } break;
  80. case NCDVALUE_VAR: {
  81. free(o->var_name);
  82. } break;
  83. default:
  84. ASSERT(0);
  85. }
  86. }
  87. int NCDValue_Type (NCDValue *o)
  88. {
  89. value_assert(o);
  90. return o->type;
  91. }
  92. int NCDValue_InitString (NCDValue *o, const char *str)
  93. {
  94. return NCDValue_InitStringBin(o, (const uint8_t *)str, strlen(str));
  95. }
  96. int NCDValue_InitStringBin (NCDValue *o, const uint8_t *str, size_t len)
  97. {
  98. if (len == SIZE_MAX) {
  99. return 0;
  100. }
  101. if (!(o->string = malloc(len + 1))) {
  102. return 0;
  103. }
  104. memcpy(o->string, str, len);
  105. o->string[len] = '\0';
  106. o->string_len = len;
  107. o->type = NCDVALUE_STRING;
  108. return 1;
  109. }
  110. char * NCDValue_StringValue (NCDValue *o)
  111. {
  112. ASSERT(o->type == NCDVALUE_STRING)
  113. return (char *)o->string;
  114. }
  115. size_t NCDValue_StringLength (NCDValue *o)
  116. {
  117. ASSERT(o->type == NCDVALUE_STRING)
  118. return o->string_len;
  119. }
  120. void NCDValue_InitList (NCDValue *o)
  121. {
  122. o->type = NCDVALUE_LIST;
  123. LinkedList1_Init(&o->list);
  124. o->list_count = 0;
  125. }
  126. size_t NCDValue_ListCount (NCDValue *o)
  127. {
  128. value_assert(o);
  129. ASSERT(o->type == NCDVALUE_LIST)
  130. return o->list_count;
  131. }
  132. int NCDValue_ListAppend (NCDValue *o, NCDValue v)
  133. {
  134. value_assert(o);
  135. ASSERT(o->type == NCDVALUE_LIST)
  136. value_assert(&v);
  137. if (o->list_count == SIZE_MAX) {
  138. return 0;
  139. }
  140. struct NCDValue__list_element *e = malloc(sizeof(*e));
  141. if (!e) {
  142. return 0;
  143. }
  144. e->v = v;
  145. LinkedList1_Append(&o->list, &e->list_node);
  146. o->list_count++;
  147. return 1;
  148. }
  149. int NCDValue_ListPrepend (NCDValue *o, NCDValue v)
  150. {
  151. value_assert(o);
  152. ASSERT(o->type == NCDVALUE_LIST)
  153. value_assert(&v);
  154. if (o->list_count == SIZE_MAX) {
  155. return 0;
  156. }
  157. struct NCDValue__list_element *e = malloc(sizeof(*e));
  158. if (!e) {
  159. return 0;
  160. }
  161. e->v = v;
  162. LinkedList1_Prepend(&o->list, &e->list_node);
  163. o->list_count++;
  164. return 1;
  165. }
  166. NCDValue * NCDValue_ListFirst (NCDValue *o)
  167. {
  168. value_assert(o);
  169. ASSERT(o->type == NCDVALUE_LIST)
  170. LinkedList1Node *ln = LinkedList1_GetFirst(&o->list);
  171. if (!ln) {
  172. return NULL;
  173. }
  174. struct NCDValue__list_element *e = UPPER_OBJECT(ln, struct NCDValue__list_element, list_node);
  175. return &e->v;
  176. }
  177. NCDValue * NCDValue_ListNext (NCDValue *o, NCDValue *ev)
  178. {
  179. value_assert(o);
  180. ASSERT(o->type == NCDVALUE_LIST)
  181. struct NCDValue__list_element *cur_e = UPPER_OBJECT(ev, struct NCDValue__list_element, v);
  182. LinkedList1Node *ln = LinkedList1Node_Next(&cur_e->list_node);
  183. if (!ln) {
  184. return NULL;
  185. }
  186. struct NCDValue__list_element *e = UPPER_OBJECT(ln, struct NCDValue__list_element, list_node);
  187. return &e->v;
  188. }
  189. void NCDValue_InitMap (NCDValue *o)
  190. {
  191. o->type = NCDVALUE_MAP;
  192. LinkedList1_Init(&o->map_list);
  193. o->map_count = 0;
  194. }
  195. size_t NCDValue_MapCount (NCDValue *o)
  196. {
  197. value_assert(o);
  198. ASSERT(o->type == NCDVALUE_MAP)
  199. return o->map_count;
  200. }
  201. NCDValue * NCDValue_MapPrepend (NCDValue *o, NCDValue key, NCDValue val)
  202. {
  203. value_assert(o);
  204. ASSERT(o->type == NCDVALUE_MAP)
  205. value_assert(&key);
  206. value_assert(&val);
  207. if (o->map_count == SIZE_MAX) {
  208. return NULL;
  209. }
  210. struct NCDValue__map_element *e = malloc(sizeof(*e));
  211. if (!e) {
  212. return NULL;
  213. }
  214. e->key = key;
  215. e->val = val;
  216. LinkedList1_Prepend(&o->map_list, &e->list_node);
  217. o->map_count++;
  218. return &e->key;
  219. }
  220. NCDValue * NCDValue_MapFirstKey (NCDValue *o)
  221. {
  222. value_assert(o);
  223. ASSERT(o->type == NCDVALUE_MAP)
  224. LinkedList1Node *ln = LinkedList1_GetFirst(&o->map_list);
  225. if (!ln) {
  226. return NULL;
  227. }
  228. struct NCDValue__map_element *e = UPPER_OBJECT(ln, struct NCDValue__map_element, list_node);
  229. value_assert(&e->key);
  230. value_assert(&e->val);
  231. return &e->key;
  232. }
  233. NCDValue * NCDValue_MapNextKey (NCDValue *o, NCDValue *ekey)
  234. {
  235. value_assert(o);
  236. ASSERT(o->type == NCDVALUE_MAP)
  237. struct NCDValue__map_element *e0 = UPPER_OBJECT(ekey, struct NCDValue__map_element, key);
  238. value_assert(&e0->key);
  239. value_assert(&e0->val);
  240. LinkedList1Node *ln = LinkedList1Node_Next(&e0->list_node);
  241. if (!ln) {
  242. return NULL;
  243. }
  244. struct NCDValue__map_element *e = UPPER_OBJECT(ln, struct NCDValue__map_element, list_node);
  245. value_assert(&e->key);
  246. value_assert(&e->val);
  247. return &e->key;
  248. }
  249. NCDValue * NCDValue_MapKeyValue (NCDValue *o, NCDValue *ekey)
  250. {
  251. value_assert(o);
  252. ASSERT(o->type == NCDVALUE_MAP)
  253. struct NCDValue__map_element *e = UPPER_OBJECT(ekey, struct NCDValue__map_element, key);
  254. value_assert(&e->key);
  255. value_assert(&e->val);
  256. return &e->val;
  257. }
  258. int NCDValue_InitVar (NCDValue *o, const char *var_name)
  259. {
  260. ASSERT(var_name)
  261. if (!(o->var_name = strdup(var_name))) {
  262. return 0;
  263. }
  264. o->type = NCDVALUE_VAR;
  265. return 1;
  266. }
  267. const char * NCDValue_VarName (NCDValue *o)
  268. {
  269. value_assert(o);
  270. ASSERT(o->type == NCDVALUE_VAR)
  271. return o->var_name;
  272. }
  273. void NCDProgram_Init (NCDProgram *o)
  274. {
  275. LinkedList1_Init(&o->processes_list);
  276. o->num_processes = 0;
  277. }
  278. void NCDProgram_Free (NCDProgram *o)
  279. {
  280. LinkedList1Node *ln;
  281. while (ln = LinkedList1_GetFirst(&o->processes_list)) {
  282. struct ProgramProcess *e = UPPER_OBJECT(ln, struct ProgramProcess, processes_list_node);
  283. NCDProcess_Free(&e->p);
  284. LinkedList1_Remove(&o->processes_list, &e->processes_list_node);
  285. free(e);
  286. }
  287. }
  288. NCDProcess * NCDProgram_PrependProcess (NCDProgram *o, NCDProcess p)
  289. {
  290. if (o->num_processes == SIZE_MAX) {
  291. return NULL;
  292. }
  293. struct ProgramProcess *e = malloc(sizeof(*e));
  294. if (!e) {
  295. return NULL;
  296. }
  297. LinkedList1_Prepend(&o->processes_list, &e->processes_list_node);
  298. e->p = p;
  299. o->num_processes++;
  300. return &e->p;
  301. }
  302. NCDProcess * NCDProgram_FirstProcess (NCDProgram *o)
  303. {
  304. LinkedList1Node *ln = LinkedList1_GetFirst(&o->processes_list);
  305. if (!ln) {
  306. return NULL;
  307. }
  308. struct ProgramProcess *e = UPPER_OBJECT(ln, struct ProgramProcess, processes_list_node);
  309. return &e->p;
  310. }
  311. NCDProcess * NCDProgram_NextProcess (NCDProgram *o, NCDProcess *ep)
  312. {
  313. ASSERT(ep)
  314. struct ProgramProcess *cur_e = UPPER_OBJECT(ep, struct ProgramProcess, p);
  315. LinkedList1Node *ln = LinkedList1Node_Next(&cur_e->processes_list_node);
  316. if (!ln) {
  317. return NULL;
  318. }
  319. struct ProgramProcess *e = UPPER_OBJECT(ln, struct ProgramProcess, processes_list_node);
  320. return &e->p;
  321. }
  322. size_t NCDProgram_NumProcesses (NCDProgram *o)
  323. {
  324. return o->num_processes;
  325. }
  326. int NCDProcess_Init (NCDProcess *o, int is_template, const char *name, NCDBlock block)
  327. {
  328. ASSERT(is_template == !!is_template)
  329. ASSERT(name)
  330. if (!(o->name = strdup(name))) {
  331. return 0;
  332. }
  333. o->is_template = is_template;
  334. o->block = block;
  335. return 1;
  336. }
  337. void NCDProcess_Free (NCDProcess *o)
  338. {
  339. NCDBlock_Free(&o->block);
  340. free(o->name);
  341. }
  342. int NCDProcess_IsTemplate (NCDProcess *o)
  343. {
  344. return o->is_template;
  345. }
  346. const char * NCDProcess_Name (NCDProcess *o)
  347. {
  348. return o->name;
  349. }
  350. NCDBlock * NCDProcess_Block (NCDProcess *o)
  351. {
  352. return &o->block;
  353. }
  354. void NCDBlock_Init (NCDBlock *o)
  355. {
  356. LinkedList1_Init(&o->statements_list);
  357. o->count = 0;
  358. }
  359. void NCDBlock_Free (NCDBlock *o)
  360. {
  361. LinkedList1Node *ln;
  362. while (ln = LinkedList1_GetFirst(&o->statements_list)) {
  363. struct BlockStatement *e = UPPER_OBJECT(ln, struct BlockStatement, statements_list_node);
  364. NCDStatement_Free(&e->s);
  365. LinkedList1_Remove(&o->statements_list, &e->statements_list_node);
  366. free(e);
  367. }
  368. }
  369. int NCDBlock_PrependStatement (NCDBlock *o, NCDStatement s)
  370. {
  371. return NCDBlock_InsertStatementAfter(o, NULL, s);
  372. }
  373. int NCDBlock_InsertStatementAfter (NCDBlock *o, NCDStatement *after, NCDStatement s)
  374. {
  375. struct BlockStatement *after_e = NULL;
  376. if (after) {
  377. after_e = UPPER_OBJECT(after, struct BlockStatement, s);
  378. }
  379. if (o->count == SIZE_MAX) {
  380. return 0;
  381. }
  382. struct BlockStatement *e = malloc(sizeof(*e));
  383. if (!e) {
  384. return 0;
  385. }
  386. if (after_e) {
  387. LinkedList1_InsertAfter(&o->statements_list, &e->statements_list_node, &after_e->statements_list_node);
  388. } else {
  389. LinkedList1_Prepend(&o->statements_list, &e->statements_list_node);
  390. }
  391. e->s = s;
  392. o->count++;
  393. return 1;
  394. }
  395. NCDStatement * NCDBlock_ReplaceStatement (NCDBlock *o, NCDStatement *es, NCDStatement s)
  396. {
  397. ASSERT(es)
  398. struct BlockStatement *e = UPPER_OBJECT(es, struct BlockStatement, s);
  399. NCDStatement_Free(&e->s);
  400. e->s = s;
  401. return &e->s;
  402. }
  403. NCDStatement * NCDBlock_FirstStatement (NCDBlock *o)
  404. {
  405. LinkedList1Node *ln = LinkedList1_GetFirst(&o->statements_list);
  406. if (!ln) {
  407. return NULL;
  408. }
  409. struct BlockStatement *e = UPPER_OBJECT(ln, struct BlockStatement, statements_list_node);
  410. return &e->s;
  411. }
  412. NCDStatement * NCDBlock_NextStatement (NCDBlock *o, NCDStatement *es)
  413. {
  414. ASSERT(es)
  415. struct BlockStatement *cur_e = UPPER_OBJECT(es, struct BlockStatement, s);
  416. LinkedList1Node *ln = LinkedList1Node_Next(&cur_e->statements_list_node);
  417. if (!ln) {
  418. return NULL;
  419. }
  420. struct BlockStatement *e = UPPER_OBJECT(ln, struct BlockStatement, statements_list_node);
  421. return &e->s;
  422. }
  423. size_t NCDBlock_NumStatements (NCDBlock *o)
  424. {
  425. return o->count;
  426. }
  427. int NCDStatement_InitReg (NCDStatement *o, const char *name, const char *objname, const char *cmdname, NCDValue args)
  428. {
  429. ASSERT(cmdname)
  430. ASSERT(NCDValue_Type(&args) == NCDVALUE_LIST)
  431. o->name = NULL;
  432. o->reg.objname = NULL;
  433. o->reg.cmdname = NULL;
  434. if (name && !(o->name = strdup(name))) {
  435. goto fail;
  436. }
  437. if (objname && !(o->reg.objname = strdup(objname))) {
  438. goto fail;
  439. }
  440. if (!(o->reg.cmdname = strdup(cmdname))) {
  441. goto fail;
  442. }
  443. o->type = NCDSTATEMENT_REG;
  444. o->reg.args = args;
  445. return 1;
  446. fail:
  447. free(o->name);
  448. free(o->reg.objname);
  449. free(o->reg.cmdname);
  450. return 0;
  451. }
  452. int NCDStatement_InitIf (NCDStatement *o, const char *name, NCDIfBlock ifblock)
  453. {
  454. o->name = NULL;
  455. if (name && !(o->name = strdup(name))) {
  456. return 0;
  457. }
  458. o->type = NCDSTATEMENT_IF;
  459. o->ifc.ifblock = ifblock;
  460. o->ifc.have_else = 0;
  461. return 1;
  462. }
  463. int NCDStatement_InitForeach (NCDStatement *o, const char *name, NCDValue collection, const char *name1, const char *name2, NCDBlock block)
  464. {
  465. ASSERT(name1)
  466. o->name = NULL;
  467. o->foreach.name1 = NULL;
  468. o->foreach.name2 = NULL;
  469. if (name && !(o->name = strdup(name))) {
  470. goto fail;
  471. }
  472. if (!(o->foreach.name1 = strdup(name1))) {
  473. goto fail;
  474. }
  475. if (name2 && !(o->foreach.name2 = strdup(name2))) {
  476. goto fail;
  477. }
  478. o->type = NCDSTATEMENT_FOREACH;
  479. o->foreach.collection = collection;
  480. o->foreach.block = block;
  481. o->foreach.is_grabbed = 0;
  482. return 1;
  483. fail:
  484. free(o->name);
  485. free(o->foreach.name1);
  486. free(o->foreach.name2);
  487. return 0;
  488. }
  489. void NCDStatement_Free (NCDStatement *o)
  490. {
  491. switch (o->type) {
  492. case NCDSTATEMENT_REG: {
  493. NCDValue_Free(&o->reg.args);
  494. free(o->reg.cmdname);
  495. free(o->reg.objname);
  496. } break;
  497. case NCDSTATEMENT_IF: {
  498. if (o->ifc.have_else) {
  499. NCDBlock_Free(&o->ifc.else_block);
  500. }
  501. NCDIfBlock_Free(&o->ifc.ifblock);
  502. } break;
  503. case NCDSTATEMENT_FOREACH: {
  504. if (!o->foreach.is_grabbed) {
  505. NCDBlock_Free(&o->foreach.block);
  506. NCDValue_Free(&o->foreach.collection);
  507. }
  508. free(o->foreach.name2);
  509. free(o->foreach.name1);
  510. } break;
  511. default: ASSERT(0);
  512. }
  513. free(o->name);
  514. }
  515. int NCDStatement_Type (NCDStatement *o)
  516. {
  517. return o->type;
  518. }
  519. const char * NCDStatement_Name (NCDStatement *o)
  520. {
  521. return o->name;
  522. }
  523. const char * NCDStatement_RegObjName (NCDStatement *o)
  524. {
  525. ASSERT(o->type == NCDSTATEMENT_REG)
  526. return o->reg.objname;
  527. }
  528. const char * NCDStatement_RegCmdName (NCDStatement *o)
  529. {
  530. ASSERT(o->type == NCDSTATEMENT_REG)
  531. return o->reg.cmdname;
  532. }
  533. NCDValue * NCDStatement_RegArgs (NCDStatement *o)
  534. {
  535. ASSERT(o->type == NCDSTATEMENT_REG)
  536. return &o->reg.args;
  537. }
  538. NCDIfBlock * NCDStatement_IfBlock (NCDStatement *o)
  539. {
  540. ASSERT(o->type == NCDSTATEMENT_IF)
  541. return &o->ifc.ifblock;
  542. }
  543. void NCDStatement_IfAddElse (NCDStatement *o, NCDBlock else_block)
  544. {
  545. ASSERT(o->type == NCDSTATEMENT_IF)
  546. ASSERT(!o->ifc.have_else)
  547. o->ifc.have_else = 1;
  548. o->ifc.else_block = else_block;
  549. }
  550. NCDBlock * NCDStatement_IfElse (NCDStatement *o)
  551. {
  552. ASSERT(o->type == NCDSTATEMENT_IF)
  553. if (!o->ifc.have_else) {
  554. return NULL;
  555. }
  556. return &o->ifc.else_block;
  557. }
  558. NCDBlock NCDStatement_IfGrabElse (NCDStatement *o)
  559. {
  560. ASSERT(o->type == NCDSTATEMENT_IF)
  561. ASSERT(o->ifc.have_else)
  562. o->ifc.have_else = 0;
  563. return o->ifc.else_block;
  564. }
  565. NCDValue * NCDStatement_ForeachCollection (NCDStatement *o)
  566. {
  567. ASSERT(o->type == NCDSTATEMENT_FOREACH)
  568. ASSERT(!o->foreach.is_grabbed)
  569. return &o->foreach.collection;
  570. }
  571. const char * NCDStatement_ForeachName1 (NCDStatement *o)
  572. {
  573. ASSERT(o->type == NCDSTATEMENT_FOREACH)
  574. return o->foreach.name1;
  575. }
  576. const char * NCDStatement_ForeachName2 (NCDStatement *o)
  577. {
  578. ASSERT(o->type == NCDSTATEMENT_FOREACH)
  579. return o->foreach.name2;
  580. }
  581. NCDBlock * NCDStatement_ForeachBlock (NCDStatement *o)
  582. {
  583. ASSERT(o->type == NCDSTATEMENT_FOREACH)
  584. ASSERT(!o->foreach.is_grabbed)
  585. return &o->foreach.block;
  586. }
  587. void NCDStatement_ForeachGrab (NCDStatement *o, NCDValue *out_collection, NCDBlock *out_block)
  588. {
  589. ASSERT(o->type == NCDSTATEMENT_FOREACH)
  590. ASSERT(!o->foreach.is_grabbed)
  591. *out_collection = o->foreach.collection;
  592. *out_block = o->foreach.block;
  593. o->foreach.is_grabbed = 1;
  594. }
  595. void NCDIfBlock_Init (NCDIfBlock *o)
  596. {
  597. LinkedList1_Init(&o->ifs_list);
  598. }
  599. void NCDIfBlock_Free (NCDIfBlock *o)
  600. {
  601. LinkedList1Node *ln;
  602. while (ln = LinkedList1_GetFirst(&o->ifs_list)) {
  603. struct IfBlockIf *e = UPPER_OBJECT(ln, struct IfBlockIf, ifs_list_node);
  604. NCDIf_Free(&e->ifc);
  605. LinkedList1_Remove(&o->ifs_list, &e->ifs_list_node);
  606. free(e);
  607. }
  608. }
  609. int NCDIfBlock_PrependIf (NCDIfBlock *o, NCDIf ifc)
  610. {
  611. struct IfBlockIf *e = malloc(sizeof(*e));
  612. if (!e) {
  613. return 0;
  614. }
  615. LinkedList1_Prepend(&o->ifs_list, &e->ifs_list_node);
  616. e->ifc = ifc;
  617. return 1;
  618. }
  619. NCDIf * NCDIfBlock_FirstIf (NCDIfBlock *o)
  620. {
  621. LinkedList1Node *ln = LinkedList1_GetFirst(&o->ifs_list);
  622. if (!ln) {
  623. return NULL;
  624. }
  625. struct IfBlockIf *e = UPPER_OBJECT(ln, struct IfBlockIf, ifs_list_node);
  626. return &e->ifc;
  627. }
  628. NCDIf * NCDIfBlock_NextIf (NCDIfBlock *o, NCDIf *ei)
  629. {
  630. ASSERT(ei)
  631. struct IfBlockIf *cur_e = UPPER_OBJECT(ei, struct IfBlockIf, ifc);
  632. LinkedList1Node *ln = LinkedList1Node_Next(&cur_e->ifs_list_node);
  633. if (!ln) {
  634. return NULL;
  635. }
  636. struct IfBlockIf *e = UPPER_OBJECT(ln, struct IfBlockIf, ifs_list_node);
  637. return &e->ifc;
  638. }
  639. NCDIf NCDIfBlock_GrabIf (NCDIfBlock *o, NCDIf *ei)
  640. {
  641. ASSERT(ei)
  642. struct IfBlockIf *e = UPPER_OBJECT(ei, struct IfBlockIf, ifc);
  643. NCDIf old_ifc = e->ifc;
  644. LinkedList1_Remove(&o->ifs_list, &e->ifs_list_node);
  645. free(e);
  646. return old_ifc;
  647. }
  648. void NCDIf_Init (NCDIf *o, NCDValue cond, NCDBlock block)
  649. {
  650. o->cond = cond;
  651. o->block = block;
  652. }
  653. void NCDIf_Free (NCDIf *o)
  654. {
  655. NCDValue_Free(&o->cond);
  656. NCDBlock_Free(&o->block);
  657. }
  658. void NCDIf_FreeGrab (NCDIf *o, NCDValue *out_cond, NCDBlock *out_block)
  659. {
  660. *out_cond = o->cond;
  661. *out_block = o->block;
  662. }
  663. NCDValue * NCDIf_Cond (NCDIf *o)
  664. {
  665. return &o->cond;
  666. }
  667. NCDBlock * NCDIf_Block (NCDIf *o)
  668. {
  669. return &o->block;
  670. }