cmark

My personal build of CMark ✏️

File Name Size Mode
node.c 17416B -rw-r--r--
  1 #include <stdlib.h>
  2 #include <string.h>
  3 
  4 #include "config.h"
  5 #include "node.h"
  6 
  7 static void S_node_unlink(cmark_node *node);
  8 
  9 static CMARK_INLINE bool S_is_block(cmark_node *node) {
 10   if (node == NULL) {
 11     return false;
 12   }
 13   return node->type >= CMARK_NODE_FIRST_BLOCK &&
 14          node->type <= CMARK_NODE_LAST_BLOCK;
 15 }
 16 
 17 static CMARK_INLINE bool S_is_inline(cmark_node *node) {
 18   if (node == NULL) {
 19     return false;
 20   }
 21   return node->type >= CMARK_NODE_FIRST_INLINE &&
 22          node->type <= CMARK_NODE_LAST_INLINE;
 23 }
 24 
 25 static bool S_can_contain(cmark_node *node, cmark_node *child) {
 26   if (node == NULL || child == NULL || node == child) {
 27     return false;
 28   }
 29 
 30   // Verify that child is not an ancestor of node.
 31   if (child->first_child != NULL) {
 32     cmark_node *cur = node->parent;
 33 
 34     while (cur != NULL) {
 35       if (cur == child) {
 36         return false;
 37       }
 38       cur = cur->parent;
 39     }
 40   }
 41 
 42   if (child->type == CMARK_NODE_DOCUMENT) {
 43     return false;
 44   }
 45 
 46   switch (node->type) {
 47   case CMARK_NODE_DOCUMENT:
 48   case CMARK_NODE_BLOCK_QUOTE:
 49   case CMARK_NODE_ITEM:
 50     return S_is_block(child) && child->type != CMARK_NODE_ITEM;
 51 
 52   case CMARK_NODE_LIST:
 53     return child->type == CMARK_NODE_ITEM;
 54 
 55   case CMARK_NODE_CUSTOM_BLOCK:
 56     return true;
 57 
 58   case CMARK_NODE_PARAGRAPH:
 59   case CMARK_NODE_HEADING:
 60   case CMARK_NODE_EMPH:
 61   case CMARK_NODE_STRONG:
 62   case CMARK_NODE_LINK:
 63   case CMARK_NODE_IMAGE:
 64   case CMARK_NODE_CUSTOM_INLINE:
 65     return S_is_inline(child);
 66 
 67   default:
 68     break;
 69   }
 70 
 71   return false;
 72 }
 73 
 74 cmark_node *cmark_node_new_with_mem(cmark_node_type type, cmark_mem *mem) {
 75   cmark_node *node = (cmark_node *)mem->calloc(1, sizeof(*node));
 76   node->mem = mem;
 77   node->type = (uint16_t)type;
 78 
 79   switch (node->type) {
 80   case CMARK_NODE_HEADING:
 81     node->as.heading.level = 1;
 82     break;
 83 
 84   case CMARK_NODE_LIST: {
 85     cmark_list *list = &node->as.list;
 86     list->list_type = CMARK_BULLET_LIST;
 87     list->start = 0;
 88     list->tight = false;
 89     break;
 90   }
 91 
 92   default:
 93     break;
 94   }
 95 
 96   return node;
 97 }
 98 
 99 cmark_node *cmark_node_new(cmark_node_type type) {
100   extern cmark_mem DEFAULT_MEM_ALLOCATOR;
101   return cmark_node_new_with_mem(type, &DEFAULT_MEM_ALLOCATOR);
102 }
103 
104 // Free a cmark_node list and any children.
105 static void S_free_nodes(cmark_node *e) {
106   cmark_mem *mem = e->mem;
107   cmark_node *next;
108   while (e != NULL) {
109     switch (e->type) {
110     case CMARK_NODE_CODE_BLOCK:
111       mem->free(e->data);
112       mem->free(e->as.code.info);
113       break;
114     case CMARK_NODE_TEXT:
115     case CMARK_NODE_HTML_INLINE:
116     case CMARK_NODE_CODE:
117     case CMARK_NODE_HTML_BLOCK:
118       mem->free(e->data);
119       break;
120     case CMARK_NODE_LINK:
121     case CMARK_NODE_IMAGE:
122       mem->free(e->as.link.url);
123       mem->free(e->as.link.title);
124       break;
125     case CMARK_NODE_CUSTOM_BLOCK:
126     case CMARK_NODE_CUSTOM_INLINE:
127       mem->free(e->as.custom.on_enter);
128       mem->free(e->as.custom.on_exit);
129       break;
130     default:
131       break;
132     }
133     if (e->last_child) {
134       // Splice children into list
135       e->last_child->next = e->next;
136       e->next = e->first_child;
137     }
138     next = e->next;
139     mem->free(e);
140     e = next;
141   }
142 }
143 
144 void cmark_node_free(cmark_node *node) {
145   S_node_unlink(node);
146   node->next = NULL;
147   S_free_nodes(node);
148 }
149 
150 cmark_node_type cmark_node_get_type(cmark_node *node) {
151   if (node == NULL) {
152     return CMARK_NODE_NONE;
153   } else {
154     return (cmark_node_type)node->type;
155   }
156 }
157 
158 const char *cmark_node_get_type_string(cmark_node *node) {
159   if (node == NULL) {
160     return "NONE";
161   }
162 
163   switch (node->type) {
164   case CMARK_NODE_NONE:
165     return "none";
166   case CMARK_NODE_DOCUMENT:
167     return "document";
168   case CMARK_NODE_BLOCK_QUOTE:
169     return "block_quote";
170   case CMARK_NODE_LIST:
171     return "list";
172   case CMARK_NODE_ITEM:
173     return "item";
174   case CMARK_NODE_CODE_BLOCK:
175     return "code_block";
176   case CMARK_NODE_HTML_BLOCK:
177     return "html_block";
178   case CMARK_NODE_CUSTOM_BLOCK:
179     return "custom_block";
180   case CMARK_NODE_PARAGRAPH:
181     return "paragraph";
182   case CMARK_NODE_HEADING:
183     return "heading";
184   case CMARK_NODE_THEMATIC_BREAK:
185     return "thematic_break";
186   case CMARK_NODE_TEXT:
187     return "text";
188   case CMARK_NODE_SOFTBREAK:
189     return "softbreak";
190   case CMARK_NODE_LINEBREAK:
191     return "linebreak";
192   case CMARK_NODE_CODE:
193     return "code";
194   case CMARK_NODE_HTML_INLINE:
195     return "html_inline";
196   case CMARK_NODE_CUSTOM_INLINE:
197     return "custom_inline";
198   case CMARK_NODE_EMPH:
199     return "emph";
200   case CMARK_NODE_STRONG:
201     return "strong";
202   case CMARK_NODE_LINK:
203     return "link";
204   case CMARK_NODE_IMAGE:
205     return "image";
206   }
207 
208   return "<unknown>";
209 }
210 
211 cmark_node *cmark_node_next(cmark_node *node) {
212   if (node == NULL) {
213     return NULL;
214   } else {
215     return node->next;
216   }
217 }
218 
219 cmark_node *cmark_node_previous(cmark_node *node) {
220   if (node == NULL) {
221     return NULL;
222   } else {
223     return node->prev;
224   }
225 }
226 
227 cmark_node *cmark_node_parent(cmark_node *node) {
228   if (node == NULL) {
229     return NULL;
230   } else {
231     return node->parent;
232   }
233 }
234 
235 cmark_node *cmark_node_first_child(cmark_node *node) {
236   if (node == NULL) {
237     return NULL;
238   } else {
239     return node->first_child;
240   }
241 }
242 
243 cmark_node *cmark_node_last_child(cmark_node *node) {
244   if (node == NULL) {
245     return NULL;
246   } else {
247     return node->last_child;
248   }
249 }
250 
251 static bufsize_t cmark_set_cstr(cmark_mem *mem, unsigned char **dst,
252                                 const char *src) {
253   unsigned char *old = *dst;
254   bufsize_t len;
255 
256   if (src && src[0]) {
257       len = (bufsize_t)strlen(src);
258       *dst = (unsigned char *)mem->realloc(NULL, len + 1);
259       memcpy(*dst, src, len + 1);
260   } else {
261       len = 0;
262       *dst = NULL;
263   }
264   if (old) {
265     mem->free(old);
266   }
267 
268   return len;
269 }
270 
271 void *cmark_node_get_user_data(cmark_node *node) {
272   if (node == NULL) {
273     return NULL;
274   } else {
275     return node->user_data;
276   }
277 }
278 
279 int cmark_node_set_user_data(cmark_node *node, void *user_data) {
280   if (node == NULL) {
281     return 0;
282   }
283   node->user_data = user_data;
284   return 1;
285 }
286 
287 const char *cmark_node_get_literal(cmark_node *node) {
288   if (node == NULL) {
289     return NULL;
290   }
291 
292   switch (node->type) {
293   case CMARK_NODE_HTML_BLOCK:
294   case CMARK_NODE_TEXT:
295   case CMARK_NODE_HTML_INLINE:
296   case CMARK_NODE_CODE:
297   case CMARK_NODE_CODE_BLOCK:
298     return node->data ? (char *)node->data : "";
299 
300   default:
301     break;
302   }
303 
304   return NULL;
305 }
306 
307 int cmark_node_set_literal(cmark_node *node, const char *content) {
308   if (node == NULL) {
309     return 0;
310   }
311 
312   switch (node->type) {
313   case CMARK_NODE_HTML_BLOCK:
314   case CMARK_NODE_TEXT:
315   case CMARK_NODE_HTML_INLINE:
316   case CMARK_NODE_CODE:
317   case CMARK_NODE_CODE_BLOCK:
318     node->len = cmark_set_cstr(node->mem, &node->data, content);
319     return 1;
320 
321   default:
322     break;
323   }
324 
325   return 0;
326 }
327 
328 int cmark_node_get_heading_level(cmark_node *node) {
329   if (node == NULL) {
330     return 0;
331   }
332 
333   switch (node->type) {
334   case CMARK_NODE_HEADING:
335     return node->as.heading.level;
336 
337   default:
338     break;
339   }
340 
341   return 0;
342 }
343 
344 int cmark_node_set_heading_level(cmark_node *node, int level) {
345   if (node == NULL || level < 1 || level > 6) {
346     return 0;
347   }
348 
349   switch (node->type) {
350   case CMARK_NODE_HEADING:
351     node->as.heading.level = level;
352     return 1;
353 
354   default:
355     break;
356   }
357 
358   return 0;
359 }
360 
361 cmark_list_type cmark_node_get_list_type(cmark_node *node) {
362   if (node == NULL) {
363     return CMARK_NO_LIST;
364   }
365 
366   if (node->type == CMARK_NODE_LIST) {
367     return (cmark_list_type)node->as.list.list_type;
368   } else {
369     return CMARK_NO_LIST;
370   }
371 }
372 
373 int cmark_node_set_list_type(cmark_node *node, cmark_list_type type) {
374   if (!(type == CMARK_BULLET_LIST || type == CMARK_ORDERED_LIST)) {
375     return 0;
376   }
377 
378   if (node == NULL) {
379     return 0;
380   }
381 
382   if (node->type == CMARK_NODE_LIST) {
383     node->as.list.list_type = (unsigned char)type;
384     return 1;
385   } else {
386     return 0;
387   }
388 }
389 
390 cmark_delim_type cmark_node_get_list_delim(cmark_node *node) {
391   if (node == NULL) {
392     return CMARK_NO_DELIM;
393   }
394 
395   if (node->type == CMARK_NODE_LIST) {
396     return (cmark_delim_type)node->as.list.delimiter;
397   } else {
398     return CMARK_NO_DELIM;
399   }
400 }
401 
402 int cmark_node_set_list_delim(cmark_node *node, cmark_delim_type delim) {
403   if (!(delim == CMARK_PERIOD_DELIM || delim == CMARK_PAREN_DELIM)) {
404     return 0;
405   }
406 
407   if (node == NULL) {
408     return 0;
409   }
410 
411   if (node->type == CMARK_NODE_LIST) {
412     node->as.list.delimiter = (unsigned char)delim;
413     return 1;
414   } else {
415     return 0;
416   }
417 }
418 
419 int cmark_node_get_list_start(cmark_node *node) {
420   if (node == NULL) {
421     return 0;
422   }
423 
424   if (node->type == CMARK_NODE_LIST) {
425     return node->as.list.start;
426   } else {
427     return 0;
428   }
429 }
430 
431 int cmark_node_set_list_start(cmark_node *node, int start) {
432   if (node == NULL || start < 0) {
433     return 0;
434   }
435 
436   if (node->type == CMARK_NODE_LIST) {
437     node->as.list.start = start;
438     return 1;
439   } else {
440     return 0;
441   }
442 }
443 
444 int cmark_node_get_list_tight(cmark_node *node) {
445   if (node == NULL) {
446     return 0;
447   }
448 
449   if (node->type == CMARK_NODE_LIST) {
450     return node->as.list.tight;
451   } else {
452     return 0;
453   }
454 }
455 
456 int cmark_node_set_list_tight(cmark_node *node, int tight) {
457   if (node == NULL) {
458     return 0;
459   }
460 
461   if (node->type == CMARK_NODE_LIST) {
462     node->as.list.tight = tight == 1;
463     return 1;
464   } else {
465     return 0;
466   }
467 }
468 
469 const char *cmark_node_get_fence_info(cmark_node *node) {
470   if (node == NULL) {
471     return NULL;
472   }
473 
474   if (node->type == CMARK_NODE_CODE_BLOCK) {
475     return node->as.code.info ? (char *)node->as.code.info : "";
476   } else {
477     return NULL;
478   }
479 }
480 
481 int cmark_node_set_fence_info(cmark_node *node, const char *info) {
482   if (node == NULL) {
483     return 0;
484   }
485 
486   if (node->type == CMARK_NODE_CODE_BLOCK) {
487     cmark_set_cstr(node->mem, &node->as.code.info, info);
488     return 1;
489   } else {
490     return 0;
491   }
492 }
493 
494 const char *cmark_node_get_url(cmark_node *node) {
495   if (node == NULL) {
496     return NULL;
497   }
498 
499   switch (node->type) {
500   case CMARK_NODE_LINK:
501   case CMARK_NODE_IMAGE:
502     return node->as.link.url ? (char *)node->as.link.url : "";
503   default:
504     break;
505   }
506 
507   return NULL;
508 }
509 
510 int cmark_node_set_url(cmark_node *node, const char *url) {
511   if (node == NULL) {
512     return 0;
513   }
514 
515   switch (node->type) {
516   case CMARK_NODE_LINK:
517   case CMARK_NODE_IMAGE:
518     cmark_set_cstr(node->mem, &node->as.link.url, url);
519     return 1;
520   default:
521     break;
522   }
523 
524   return 0;
525 }
526 
527 const char *cmark_node_get_title(cmark_node *node) {
528   if (node == NULL) {
529     return NULL;
530   }
531 
532   switch (node->type) {
533   case CMARK_NODE_LINK:
534   case CMARK_NODE_IMAGE:
535     return node->as.link.title ? (char *)node->as.link.title : "";
536   default:
537     break;
538   }
539 
540   return NULL;
541 }
542 
543 int cmark_node_set_title(cmark_node *node, const char *title) {
544   if (node == NULL) {
545     return 0;
546   }
547 
548   switch (node->type) {
549   case CMARK_NODE_LINK:
550   case CMARK_NODE_IMAGE:
551     cmark_set_cstr(node->mem, &node->as.link.title, title);
552     return 1;
553   default:
554     break;
555   }
556 
557   return 0;
558 }
559 
560 const char *cmark_node_get_on_enter(cmark_node *node) {
561   if (node == NULL) {
562     return NULL;
563   }
564 
565   switch (node->type) {
566   case CMARK_NODE_CUSTOM_INLINE:
567   case CMARK_NODE_CUSTOM_BLOCK:
568     return node->as.custom.on_enter ? (char *)node->as.custom.on_enter : "";
569   default:
570     break;
571   }
572 
573   return NULL;
574 }
575 
576 int cmark_node_set_on_enter(cmark_node *node, const char *on_enter) {
577   if (node == NULL) {
578     return 0;
579   }
580 
581   switch (node->type) {
582   case CMARK_NODE_CUSTOM_INLINE:
583   case CMARK_NODE_CUSTOM_BLOCK:
584     cmark_set_cstr(node->mem, &node->as.custom.on_enter, on_enter);
585     return 1;
586   default:
587     break;
588   }
589 
590   return 0;
591 }
592 
593 const char *cmark_node_get_on_exit(cmark_node *node) {
594   if (node == NULL) {
595     return NULL;
596   }
597 
598   switch (node->type) {
599   case CMARK_NODE_CUSTOM_INLINE:
600   case CMARK_NODE_CUSTOM_BLOCK:
601     return node->as.custom.on_exit ? (char *)node->as.custom.on_exit : "";
602   default:
603     break;
604   }
605 
606   return NULL;
607 }
608 
609 int cmark_node_set_on_exit(cmark_node *node, const char *on_exit) {
610   if (node == NULL) {
611     return 0;
612   }
613 
614   switch (node->type) {
615   case CMARK_NODE_CUSTOM_INLINE:
616   case CMARK_NODE_CUSTOM_BLOCK:
617     cmark_set_cstr(node->mem, &node->as.custom.on_exit, on_exit);
618     return 1;
619   default:
620     break;
621   }
622 
623   return 0;
624 }
625 
626 int cmark_node_get_start_line(cmark_node *node) {
627   if (node == NULL) {
628     return 0;
629   }
630   return node->start_line;
631 }
632 
633 int cmark_node_get_start_column(cmark_node *node) {
634   if (node == NULL) {
635     return 0;
636   }
637   return node->start_column;
638 }
639 
640 int cmark_node_get_end_line(cmark_node *node) {
641   if (node == NULL) {
642     return 0;
643   }
644   return node->end_line;
645 }
646 
647 int cmark_node_get_end_column(cmark_node *node) {
648   if (node == NULL) {
649     return 0;
650   }
651   return node->end_column;
652 }
653 
654 // Unlink a node without adjusting its next, prev, and parent pointers.
655 static void S_node_unlink(cmark_node *node) {
656   if (node == NULL) {
657     return;
658   }
659 
660   if (node->prev) {
661     node->prev->next = node->next;
662   }
663   if (node->next) {
664     node->next->prev = node->prev;
665   }
666 
667   // Adjust first_child and last_child of parent.
668   cmark_node *parent = node->parent;
669   if (parent) {
670     if (parent->first_child == node) {
671       parent->first_child = node->next;
672     }
673     if (parent->last_child == node) {
674       parent->last_child = node->prev;
675     }
676   }
677 }
678 
679 void cmark_node_unlink(cmark_node *node) {
680   S_node_unlink(node);
681 
682   node->next = NULL;
683   node->prev = NULL;
684   node->parent = NULL;
685 }
686 
687 int cmark_node_insert_before(cmark_node *node, cmark_node *sibling) {
688   if (node == NULL || sibling == NULL) {
689     return 0;
690   }
691 
692   if (!node->parent || !S_can_contain(node->parent, sibling)) {
693     return 0;
694   }
695 
696   S_node_unlink(sibling);
697 
698   cmark_node *old_prev = node->prev;
699 
700   // Insert 'sibling' between 'old_prev' and 'node'.
701   if (old_prev) {
702     old_prev->next = sibling;
703   }
704   sibling->prev = old_prev;
705   sibling->next = node;
706   node->prev = sibling;
707 
708   // Set new parent.
709   cmark_node *parent = node->parent;
710   sibling->parent = parent;
711 
712   // Adjust first_child of parent if inserted as first child.
713   if (parent && !old_prev) {
714     parent->first_child = sibling;
715   }
716 
717   return 1;
718 }
719 
720 int cmark_node_insert_after(cmark_node *node, cmark_node *sibling) {
721   if (node == NULL || sibling == NULL) {
722     return 0;
723   }
724 
725   if (!node->parent || !S_can_contain(node->parent, sibling)) {
726     return 0;
727   }
728 
729   S_node_unlink(sibling);
730 
731   cmark_node *old_next = node->next;
732 
733   // Insert 'sibling' between 'node' and 'old_next'.
734   if (old_next) {
735     old_next->prev = sibling;
736   }
737   sibling->next = old_next;
738   sibling->prev = node;
739   node->next = sibling;
740 
741   // Set new parent.
742   cmark_node *parent = node->parent;
743   sibling->parent = parent;
744 
745   // Adjust last_child of parent if inserted as last child.
746   if (parent && !old_next) {
747     parent->last_child = sibling;
748   }
749 
750   return 1;
751 }
752 
753 int cmark_node_replace(cmark_node *oldnode, cmark_node *newnode) {
754   if (!cmark_node_insert_before(oldnode, newnode)) {
755     return 0;
756   }
757   cmark_node_unlink(oldnode);
758   return 1;
759 }
760 
761 int cmark_node_prepend_child(cmark_node *node, cmark_node *child) {
762   if (!S_can_contain(node, child)) {
763     return 0;
764   }
765 
766   S_node_unlink(child);
767 
768   cmark_node *old_first_child = node->first_child;
769 
770   child->next = old_first_child;
771   child->prev = NULL;
772   child->parent = node;
773   node->first_child = child;
774 
775   if (old_first_child) {
776     old_first_child->prev = child;
777   } else {
778     // Also set last_child if node previously had no children.
779     node->last_child = child;
780   }
781 
782   return 1;
783 }
784 
785 int cmark_node_append_child(cmark_node *node, cmark_node *child) {
786   if (!S_can_contain(node, child)) {
787     return 0;
788   }
789 
790   S_node_unlink(child);
791 
792   cmark_node *old_last_child = node->last_child;
793 
794   child->next = NULL;
795   child->prev = old_last_child;
796   child->parent = node;
797   node->last_child = child;
798 
799   if (old_last_child) {
800     old_last_child->next = child;
801   } else {
802     // Also set first_child if node previously had no children.
803     node->first_child = child;
804   }
805 
806   return 1;
807 }
808 
809 static void S_print_error(FILE *out, cmark_node *node, const char *elem) {
810   if (out == NULL) {
811     return;
812   }
813   fprintf(out, "Invalid '%s' in node type %s at %d:%d\n", elem,
814           cmark_node_get_type_string(node), node->start_line,
815           node->start_column);
816 }
817 
818 int cmark_node_check(cmark_node *node, FILE *out) {
819   cmark_node *cur;
820   int errors = 0;
821 
822   if (!node) {
823     return 0;
824   }
825 
826   cur = node;
827   for (;;) {
828     if (cur->first_child) {
829       if (cur->first_child->prev != NULL) {
830         S_print_error(out, cur->first_child, "prev");
831         cur->first_child->prev = NULL;
832         ++errors;
833       }
834       if (cur->first_child->parent != cur) {
835         S_print_error(out, cur->first_child, "parent");
836         cur->first_child->parent = cur;
837         ++errors;
838       }
839       cur = cur->first_child;
840       continue;
841     }
842 
843   next_sibling:
844     if (cur == node) {
845       break;
846     }
847     if (cur->next) {
848       if (cur->next->prev != cur) {
849         S_print_error(out, cur->next, "prev");
850         cur->next->prev = cur;
851         ++errors;
852       }
853       if (cur->next->parent != cur->parent) {
854         S_print_error(out, cur->next, "parent");
855         cur->next->parent = cur->parent;
856         ++errors;
857       }
858       cur = cur->next;
859       continue;
860     }
861 
862     if (cur->parent->last_child != cur) {
863       S_print_error(out, cur->parent, "last_child");
864       cur->parent->last_child = cur;
865       ++errors;
866     }
867     cur = cur->parent;
868     goto next_sibling;
869   }
870 
871   return errors;
872 }