1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
|
#ifndef CMARK_AST_H
#define CMARK_AST_H
#include <stdio.h>
#include "config.h"
#include "buffer.h"
#include "chunk.h"
#include "cmark.h"
#ifdef __cplusplus
extern "C" {
#endif
#define REFMAP_SIZE 16
#define MAX_LINK_LABEL_LENGTH 1000
struct cmark_node_inl {
cmark_inl_tag tag;
union {
cmark_chunk literal;
struct cmark_node_inl *inlines;
struct {
struct cmark_node_inl *label;
unsigned char *url;
unsigned char *title;
} linkable;
} content;
struct cmark_node_inl *next;
};
struct cmark_reference {
struct cmark_reference *next;
unsigned char *label;
unsigned char *url;
unsigned char *title;
unsigned int hash;
};
typedef struct cmark_reference cmark_reference;
struct cmark_reference_map {
cmark_reference *table[REFMAP_SIZE];
};
typedef struct cmark_reference_map cmark_reference_map;
// Types for blocks
struct cmark_ListData {
cmark_list_type list_type;
int marker_offset;
int padding;
int start;
cmark_delim_type delimiter;
unsigned char bullet_char;
bool tight;
};
struct cmark_FencedCodeData {
int fence_length;
int fence_offset;
unsigned char fence_char;
cmark_strbuf info;
};
struct cmark_node_block {
cmark_block_tag tag;
int start_line;
int start_column;
int end_line;
bool open;
bool last_line_blank;
struct cmark_node_block* children;
struct cmark_node_block* last_child;
struct cmark_node_block* parent;
cmark_strbuf string_content;
struct cmark_node_inl* inline_content;
union {
struct cmark_ListData list;
struct cmark_FencedCodeData code;
struct {
int level;
} header;
} as;
struct cmark_node_block *next;
struct cmark_node_block *prev;
};
struct cmark_doc_parser {
struct cmark_reference_map *refmap;
struct cmark_node_block* root;
struct cmark_node_block* current;
int line_number;
cmark_strbuf *curline;
};
unsigned char *cmark_clean_autolink(chunk *url, int is_email);
static inline cmark_node_inl *cmark_make_link(cmark_node_inl *label, unsigned char *url, unsigned char *title)
{
cmark_node_inl* e = (cmark_node_inl *)calloc(1, sizeof(*e));
if(e != NULL) {
e->tag = CMARK_INL_LINK;
e->content.linkable.label = label;
e->content.linkable.url = url;
e->content.linkable.title = title;
e->next = NULL;
}
return e;
}
static inline cmark_node_inl* cmark_make_autolink(cmark_node_inl* label, cmark_chunk url, int is_email)
{
return cmark_make_link(label, cmark_clean_autolink(&url, is_email), NULL);
}
static inline cmark_node_inl* cmark_make_inlines(cmark_inl_tag t, cmark_node_inl* contents)
{
cmark_node_inl * e = (cmark_node_inl *)calloc(1, sizeof(*e));
if(e != NULL) {
e->tag = t;
e->content.inlines = contents;
e->next = NULL;
}
return e;
}
// Create an inline with a literal string value.
static inline cmark_node_inl* cmark_make_literal(cmark_inl_tag t, cmark_chunk s)
{
cmark_node_inl * e = (cmark_node_inl *)calloc(1, sizeof(*e));
if(e != NULL) {
e->tag = t;
e->content.literal = s;
e->next = NULL;
}
return e;
}
// Create an inline with no value.
static inline cmark_node_inl* cmark_make_simple(cmark_inl_tag t)
{
cmark_node_inl* e = (cmark_node_inl *)calloc(1, sizeof(*e));
if(e != NULL) {
e->tag = t;
e->next = NULL;
}
return e;
}
// Macros for creating various kinds of simple.
#define cmark_make_str(s) cmark_make_literal(INL_STRING, s)
#define cmark_make_code(s) cmark_make_literal(INL_CODE, s)
#define cmark_make_raw_html(s) cmark_make_literal(INL_RAW_HTML, s)
#define cmark_make_linebreak() cmark_make_simple(INL_LINEBREAK)
#define cmark_make_softbreak() cmark_make_simple(INL_SOFTBREAK)
#define cmark_make_emph(contents) cmark_make_inlines(INL_EMPH, contents)
#define cmark_make_strong(contents) cmark_make_inlines(INL_STRONG, contents)
#ifndef CMARK_NO_SHORT_NAMES
#define node_inl cmark_node_inl
#define ListData cmark_ListData
#define FencedCodeData cmark_FencedCodeData
#define node_block cmark_node_block
#define make_link cmark_make_link
#define make_autolink cmark_make_autolink
#define make_str cmark_make_str
#define make_code cmark_make_code
#define make_raw_html cmark_make_raw_html
#define make_linebreak cmark_make_linebreak
#define make_softbreak cmark_make_softbreak
#define make_emph cmark_make_emph
#define make_strong cmark_make_strong
#define make_simple cmark_make_simple
#define make_literal cmark_make_literal
#define make_inlines cmark_make_inlines
#endif
#ifdef __cplusplus
}
#endif
#endif
|