xref: /petsc/src/sys/objects/optionsyaml.c (revision 7a30eb059abdcd5c3e394e47df9719ac271b12c3)
1951eb098SLisandro Dalcin #define PETSC_DESIRE_FEATURE_TEST_MACROS /* for strdup() */
2f996e368SLisandro Dalcin #include <petsc/private/petscimpl.h>     /*I  "petscsys.h"  I*/
3951eb098SLisandro Dalcin 
4f996e368SLisandro Dalcin #if defined(PETSC_HAVE_YAML)
5951eb098SLisandro Dalcin #include <yaml.h>  /* use external LibYAML */
6951eb098SLisandro Dalcin #else
7951eb098SLisandro Dalcin #include <../src/sys/yaml/include/yaml.h>
8f996e368SLisandro Dalcin #endif
9f996e368SLisandro Dalcin 
10f996e368SLisandro Dalcin static MPI_Comm petsc_yaml_comm = MPI_COMM_NULL; /* only used for parallel error handling */
11f996e368SLisandro Dalcin 
12f996e368SLisandro Dalcin PETSC_STATIC_INLINE MPI_Comm PetscYAMLGetComm(void)
13f996e368SLisandro Dalcin {
14f996e368SLisandro Dalcin   return PetscLikely(petsc_yaml_comm != MPI_COMM_NULL) ? petsc_yaml_comm : (petsc_yaml_comm = PETSC_COMM_SELF);
15f996e368SLisandro Dalcin }
16f996e368SLisandro Dalcin 
17f996e368SLisandro Dalcin PETSC_STATIC_INLINE MPI_Comm PetscYAMLSetComm(MPI_Comm comm)
18f996e368SLisandro Dalcin {
19f996e368SLisandro Dalcin   MPI_Comm prev = PetscYAMLGetComm(); petsc_yaml_comm = comm; return prev;
20f996e368SLisandro Dalcin }
21f996e368SLisandro Dalcin 
22f996e368SLisandro Dalcin #define TAG(node) ((const char *)((node)->tag))
23f996e368SLisandro Dalcin #define STR(node) ((const char *)((node)->data.scalar.value))
24f996e368SLisandro Dalcin #define SEQ(node) ((node)->data.sequence.items)
25f996e368SLisandro Dalcin #define MAP(node) ((node)->data.mapping.pairs)
26f996e368SLisandro Dalcin 
27f996e368SLisandro Dalcin static PetscErrorCode PetscParseLayerYAML(PetscOptions options, yaml_document_t *doc, yaml_node_t *node)
28f996e368SLisandro Dalcin {
29f996e368SLisandro Dalcin   MPI_Comm         comm = PetscYAMLGetComm();
30f996e368SLisandro Dalcin   char             name[PETSC_MAX_OPTION_NAME] = "", prefix[PETSC_MAX_OPTION_NAME] = "";
31f996e368SLisandro Dalcin   PetscErrorCode   ierr;
32f996e368SLisandro Dalcin 
33f996e368SLisandro Dalcin   PetscFunctionBegin;
34f996e368SLisandro Dalcin   if (node->type != YAML_MAPPING_NODE) SETERRQ(comm, PETSC_ERR_SUP, "Unsupported YAML node type: expected mapping");
35f996e368SLisandro Dalcin   for (yaml_node_pair_t *pair = MAP(node).start; pair < MAP(node).top; pair++) {
36f996e368SLisandro Dalcin     yaml_node_t *keynode = yaml_document_get_node(doc, pair->key);
37f996e368SLisandro Dalcin     yaml_node_t *valnode = yaml_document_get_node(doc, pair->value);
38f996e368SLisandro Dalcin     PetscBool   isMergeKey,isDummyKey,isIncludeTag;
39f996e368SLisandro Dalcin 
40f996e368SLisandro Dalcin     if (!keynode) SETERRQ(comm, PETSC_ERR_LIB, "Corrupt YAML document");
41f996e368SLisandro Dalcin     if (!valnode) SETERRQ(comm, PETSC_ERR_LIB, "Corrupt YAML document");
42f996e368SLisandro Dalcin     if (keynode->type != YAML_SCALAR_NODE) SETERRQ(comm, PETSC_ERR_SUP, "Unsupported YAML node type: expected scalar");
43f996e368SLisandro Dalcin 
44f996e368SLisandro Dalcin     /* "<<" is the merge key: don't increment the prefix */
45f996e368SLisandro Dalcin     ierr = PetscStrcmp(STR(keynode), "<<", &isMergeKey);CHKERRQ(ierr);
46f996e368SLisandro Dalcin     if (isMergeKey) {
47f996e368SLisandro Dalcin       if (valnode->type == YAML_SEQUENCE_NODE) {
48f996e368SLisandro Dalcin         for (yaml_node_item_t *item = SEQ(valnode).start; item < SEQ(valnode).top; item++) {
49f996e368SLisandro Dalcin           yaml_node_t *itemnode = yaml_document_get_node(doc, *item);
50f996e368SLisandro Dalcin           if (!itemnode) SETERRQ(comm, PETSC_ERR_LIB, "Corrupt YAML document");
51f996e368SLisandro Dalcin           if (itemnode->type != YAML_MAPPING_NODE) SETERRQ(comm, PETSC_ERR_SUP, "Unsupported YAML node type: expected mapping");
52f996e368SLisandro Dalcin           ierr = PetscParseLayerYAML(options, doc, itemnode);CHKERRQ(ierr);
53f996e368SLisandro Dalcin         }
54f996e368SLisandro Dalcin       } else if (valnode->type == YAML_MAPPING_NODE) {
55f996e368SLisandro Dalcin         ierr = PetscParseLayerYAML(options, doc, valnode);CHKERRQ(ierr);
56f996e368SLisandro Dalcin       } else SETERRQ(comm, PETSC_ERR_SUP, "Unsupported YAML node type: expected sequence or mapping");
57f996e368SLisandro Dalcin       continue; /* to next pair */
58f996e368SLisandro Dalcin     }
59f996e368SLisandro Dalcin 
60f996e368SLisandro Dalcin     /* "$$*" are treated as dummy keys, we use them for !include tags and to define anchors */
61f996e368SLisandro Dalcin     ierr = PetscStrbeginswith(STR(keynode), "$$", &isDummyKey);CHKERRQ(ierr);
62f996e368SLisandro Dalcin     if (isDummyKey) {
63f996e368SLisandro Dalcin       ierr = PetscStrendswith(TAG(valnode), "!include", &isIncludeTag);CHKERRQ(ierr);CHKERRQ(ierr);
64f996e368SLisandro Dalcin       if (isIncludeTag) { /* TODO: add proper support relative paths */
65f996e368SLisandro Dalcin         ierr = PetscOptionsInsertFileYAML(comm, options, STR(valnode), PETSC_TRUE);CHKERRQ(ierr);
66f996e368SLisandro Dalcin       }
67f996e368SLisandro Dalcin       continue; /* to next pair */
68f996e368SLisandro Dalcin     }
69f996e368SLisandro Dalcin 
70f996e368SLisandro Dalcin     if (valnode->type == YAML_SCALAR_NODE) {
71f996e368SLisandro Dalcin       ierr = PetscSNPrintf(name, sizeof(name), "-%s", STR(keynode));CHKERRQ(ierr);
72f996e368SLisandro Dalcin       ierr = PetscOptionsSetValue(options, name, STR(valnode));CHKERRQ(ierr);
73f996e368SLisandro Dalcin 
74f996e368SLisandro Dalcin     } else if (valnode->type == YAML_SEQUENCE_NODE) {
75f996e368SLisandro Dalcin       PetscSegBuffer seg;
76f996e368SLisandro Dalcin       char           *buf, *strlist;
77f996e368SLisandro Dalcin       PetscBool      addSep = PETSC_FALSE;
78f996e368SLisandro Dalcin 
79f996e368SLisandro Dalcin       ierr = PetscSegBufferCreate(sizeof(char), PETSC_MAX_PATH_LEN, &seg);CHKERRQ(ierr);
80f996e368SLisandro Dalcin       for (yaml_node_item_t *item = SEQ(valnode).start; item < SEQ(valnode).top; item++) {
81f996e368SLisandro Dalcin         yaml_node_t *itemnode = yaml_document_get_node(doc, *item);
82f996e368SLisandro Dalcin         const char  *itemstr = NULL;
83f996e368SLisandro Dalcin         size_t       itemlen;
84f996e368SLisandro Dalcin 
85f996e368SLisandro Dalcin         if (!itemnode) SETERRQ(comm, PETSC_ERR_LIB, "Corrupt YAML document");
86f996e368SLisandro Dalcin 
87f996e368SLisandro Dalcin         if (itemnode->type == YAML_SCALAR_NODE) {
88f996e368SLisandro Dalcin           itemstr = STR(itemnode);
89f996e368SLisandro Dalcin 
90f996e368SLisandro Dalcin         } else if (itemnode->type == YAML_MAPPING_NODE) {
91f996e368SLisandro Dalcin           yaml_node_pair_t *kvn = itemnode->data.mapping.pairs.start;
92f996e368SLisandro Dalcin           yaml_node_pair_t *top = itemnode->data.mapping.pairs.top;
93f996e368SLisandro Dalcin 
94f996e368SLisandro Dalcin           if (top - kvn > 1) SETERRQ(comm, PETSC_ERR_SUP, "Unsupported YAML node value: expected a single key:value pair");
95f996e368SLisandro Dalcin           if (top - kvn > 0) {
96f996e368SLisandro Dalcin             yaml_node_t *kn = yaml_document_get_node(doc, kvn->key);
97f996e368SLisandro Dalcin             yaml_node_t *vn = yaml_document_get_node(doc, kvn->value);
98f996e368SLisandro Dalcin 
99f996e368SLisandro Dalcin             if (!kn) SETERRQ(comm, PETSC_ERR_LIB, "Corrupt YAML document");
100f996e368SLisandro Dalcin             if (!vn) SETERRQ(comm, PETSC_ERR_LIB, "Corrupt YAML document");
101f996e368SLisandro Dalcin             if (kn->type != YAML_SCALAR_NODE) SETERRQ(comm, PETSC_ERR_SUP, "Unsupported YAML node type: expected scalar");
102f996e368SLisandro Dalcin 
103f996e368SLisandro Dalcin             ierr = PetscStrcmp(STR(kn), "<<", &isMergeKey);CHKERRQ(ierr);
104f996e368SLisandro Dalcin             if (isMergeKey) SETERRQ(comm, PETSC_ERR_SUP, "Unsupported YAML node value: merge key '<<' not supported here");
105f996e368SLisandro Dalcin 
106f996e368SLisandro Dalcin             ierr = PetscStrbeginswith(STR(kn), "$$", &isDummyKey);CHKERRQ(ierr);
107f996e368SLisandro Dalcin             if (isDummyKey) continue;
108f996e368SLisandro Dalcin             itemstr = STR(kn);
109f996e368SLisandro Dalcin           }
110f996e368SLisandro Dalcin 
111f996e368SLisandro Dalcin           ierr = PetscSNPrintf(prefix,sizeof(prefix), "%s_", STR(keynode));CHKERRQ(ierr);
112f996e368SLisandro Dalcin           ierr = PetscOptionsPrefixPush(options, prefix);CHKERRQ(ierr);
113f996e368SLisandro Dalcin           ierr = PetscParseLayerYAML(options, doc, itemnode);CHKERRQ(ierr);
114f996e368SLisandro Dalcin           ierr = PetscOptionsPrefixPop(options);CHKERRQ(ierr);
115f996e368SLisandro Dalcin 
116f996e368SLisandro Dalcin         } else SETERRQ(comm, PETSC_ERR_SUP, "Unsupported YAML node type: expected scalar or mapping");
117f996e368SLisandro Dalcin 
118f996e368SLisandro Dalcin         ierr = PetscStrlen(itemstr, &itemlen);CHKERRQ(ierr);
119f996e368SLisandro Dalcin         if (itemlen) {
120f996e368SLisandro Dalcin           if (addSep) {
121f996e368SLisandro Dalcin             ierr = PetscSegBufferGet(seg, 1, &buf);CHKERRQ(ierr);
122f996e368SLisandro Dalcin             ierr = PetscArraycpy(buf, ",", 1);CHKERRQ(ierr);
123f996e368SLisandro Dalcin           }
124f996e368SLisandro Dalcin           ierr = PetscSegBufferGet(seg, itemlen, &buf);CHKERRQ(ierr);
125f996e368SLisandro Dalcin           ierr = PetscArraycpy(buf, itemstr, itemlen);CHKERRQ(ierr);
126f996e368SLisandro Dalcin           addSep = PETSC_TRUE;
127f996e368SLisandro Dalcin         }
128f996e368SLisandro Dalcin       }
129f996e368SLisandro Dalcin       ierr = PetscSegBufferGet(seg, 1, &buf);CHKERRQ(ierr);
130f996e368SLisandro Dalcin       ierr = PetscArrayzero(buf, 1);CHKERRQ(ierr);
131f996e368SLisandro Dalcin       ierr = PetscSegBufferExtractAlloc(seg, &strlist);CHKERRQ(ierr);
132f996e368SLisandro Dalcin       ierr = PetscSegBufferDestroy(&seg);CHKERRQ(ierr);
133f996e368SLisandro Dalcin 
134f996e368SLisandro Dalcin       ierr = PetscSNPrintf(name, sizeof(name), "-%s", STR(keynode));CHKERRQ(ierr);
135f996e368SLisandro Dalcin       ierr = PetscOptionsSetValue(options, name, strlist);CHKERRQ(ierr);
136f996e368SLisandro Dalcin       ierr = PetscFree(strlist);CHKERRQ(ierr);
137f996e368SLisandro Dalcin 
138f996e368SLisandro Dalcin     } else if (valnode->type == YAML_MAPPING_NODE) {
139f996e368SLisandro Dalcin       ierr = PetscSNPrintf(prefix,sizeof(prefix), "%s_", STR(keynode));CHKERRQ(ierr);
140f996e368SLisandro Dalcin       ierr = PetscOptionsPrefixPush(options, prefix);CHKERRQ(ierr);
141f996e368SLisandro Dalcin       ierr = PetscParseLayerYAML(options, doc, valnode);CHKERRQ(ierr);
142f996e368SLisandro Dalcin       ierr = PetscOptionsPrefixPop(options);CHKERRQ(ierr);
143f996e368SLisandro Dalcin 
144f996e368SLisandro Dalcin     } else SETERRQ(comm, PETSC_ERR_SUP, "Unsupported YAML node type: expected scalar, sequence or mapping");
145f996e368SLisandro Dalcin   }
146f996e368SLisandro Dalcin   PetscFunctionReturn(0);
147f996e368SLisandro Dalcin }
148f996e368SLisandro Dalcin 
149f996e368SLisandro Dalcin /*@C
150f996e368SLisandro Dalcin    PetscOptionsInsertStringYAML - Inserts YAML-formatted options into the database from a string
151f996e368SLisandro Dalcin 
152f996e368SLisandro Dalcin    Logically Collective
153f996e368SLisandro Dalcin 
154f996e368SLisandro Dalcin    Input Parameter:
155f996e368SLisandro Dalcin +  options - options database, use NULL for default global database
156f996e368SLisandro Dalcin -  in_str - YAML-formatted string options
157f996e368SLisandro Dalcin 
158f996e368SLisandro Dalcin    Level: intermediate
159f996e368SLisandro Dalcin 
160f996e368SLisandro Dalcin .seealso: PetscOptionsSetValue(), PetscOptionsView(), PetscOptionsHasName(), PetscOptionsGetInt(),
161f996e368SLisandro Dalcin           PetscOptionsGetReal(), PetscOptionsGetString(), PetscOptionsGetIntArray(), PetscOptionsBool(),
162f996e368SLisandro Dalcin           PetscOptionsName(), PetscOptionsBegin(), PetscOptionsEnd(), PetscOptionsHead(),
163f996e368SLisandro Dalcin           PetscOptionsStringArray(),PetscOptionsRealArray(), PetscOptionsScalar(),
164f996e368SLisandro Dalcin           PetscOptionsBoolGroupBegin(), PetscOptionsBoolGroup(), PetscOptionsBoolGroupEnd(),
165f996e368SLisandro Dalcin           PetscOptionsFList(), PetscOptionsEList(), PetscOptionsInsertFile(), PetscOptionsInsertFileYAML()
166f996e368SLisandro Dalcin @*/
167f996e368SLisandro Dalcin PetscErrorCode PetscOptionsInsertStringYAML(PetscOptions options,const char in_str[])
168f996e368SLisandro Dalcin {
169f996e368SLisandro Dalcin   MPI_Comm        comm = PetscYAMLGetComm();
170f996e368SLisandro Dalcin   yaml_parser_t   parser;
171f996e368SLisandro Dalcin   yaml_document_t doc;
172f996e368SLisandro Dalcin   yaml_node_t     *root;
173f996e368SLisandro Dalcin   PetscErrorCode  ierr;
174f996e368SLisandro Dalcin 
175f996e368SLisandro Dalcin   PetscFunctionBegin;
176f996e368SLisandro Dalcin   if (!in_str) in_str = "";
177f996e368SLisandro Dalcin   ierr = !yaml_parser_initialize(&parser); if (ierr) SETERRQ(comm, PETSC_ERR_LIB, "YAML parser initialization error");
178f996e368SLisandro Dalcin   yaml_parser_set_input_string(&parser, (const unsigned char *)in_str, strlen(in_str));
179f996e368SLisandro Dalcin   do {
180f996e368SLisandro Dalcin     ierr = !yaml_parser_load(&parser, &doc); if (ierr) SETERRQ(comm, PETSC_ERR_LIB, "YAML parser loading error");
181f996e368SLisandro Dalcin     root = yaml_document_get_root_node(&doc);
182f996e368SLisandro Dalcin     if (root) {
183f996e368SLisandro Dalcin       ierr = PetscParseLayerYAML(options, &doc, root);CHKERRQ(ierr);
184f996e368SLisandro Dalcin     }
185f996e368SLisandro Dalcin     yaml_document_delete(&doc);
186f996e368SLisandro Dalcin   } while (root);
187f996e368SLisandro Dalcin   yaml_parser_delete(&parser);
188f996e368SLisandro Dalcin   PetscFunctionReturn(0);
189f996e368SLisandro Dalcin }
190f996e368SLisandro Dalcin 
191f996e368SLisandro Dalcin /*@C
192*7a30eb05SPatrick Sanan   PetscOptionsInsertFileYAML - Insert a YAML-formatted file in the options database
193f996e368SLisandro Dalcin 
194f996e368SLisandro Dalcin   Collective
195f996e368SLisandro Dalcin 
196f996e368SLisandro Dalcin   Input Parameter:
197f996e368SLisandro Dalcin +   comm - the processes that will share the options (usually PETSC_COMM_WORLD)
198f996e368SLisandro Dalcin .   options - options database, use NULL for default global database
199f996e368SLisandro Dalcin .   file - name of file
200f996e368SLisandro Dalcin -   require - if PETSC_TRUE will generate an error if the file does not exist
201f996e368SLisandro Dalcin 
202f996e368SLisandro Dalcin   PETSc will generate an error condition that stops the program if a YAML error
203f996e368SLisandro Dalcin   is detected, hence the user should check that the YAML file is valid before
204f996e368SLisandro Dalcin   supplying it, for instance at http://www.yamllint.com/ .
205f996e368SLisandro Dalcin 
206*7a30eb05SPatrick Sanan   Uses PetscOptionsInsertStringYAML().
207f996e368SLisandro Dalcin 
208f996e368SLisandro Dalcin   Level: intermediate
209f996e368SLisandro Dalcin 
210f996e368SLisandro Dalcin .seealso: PetscOptionsSetValue(), PetscOptionsView(), PetscOptionsHasName(), PetscOptionsGetInt(),
211f996e368SLisandro Dalcin           PetscOptionsGetReal(), PetscOptionsGetString(), PetscOptionsGetIntArray(), PetscOptionsBool(),
212f996e368SLisandro Dalcin           PetscOptionsName(), PetscOptionsBegin(), PetscOptionsEnd(), PetscOptionsHead(),
213f996e368SLisandro Dalcin           PetscOptionsStringArray(),PetscOptionsRealArray(), PetscOptionsScalar(),
214f996e368SLisandro Dalcin           PetscOptionsBoolGroupBegin(), PetscOptionsBoolGroup(), PetscOptionsBoolGroupEnd(),
215f996e368SLisandro Dalcin           PetscOptionsFList(), PetscOptionsEList(), PetscOptionsInsertFile(), PetscOptionsInsertStringYAML()
216f996e368SLisandro Dalcin @*/
217f996e368SLisandro Dalcin PetscErrorCode PetscOptionsInsertFileYAML(MPI_Comm comm,PetscOptions options,const char file[],PetscBool require)
218f996e368SLisandro Dalcin {
219f996e368SLisandro Dalcin   int            yamlLength = -1;
220f996e368SLisandro Dalcin   char          *yamlString = NULL;
221f996e368SLisandro Dalcin   MPI_Comm       prev;
222f996e368SLisandro Dalcin   PetscMPIInt    rank;
223f996e368SLisandro Dalcin   PetscErrorCode ierr;
224f996e368SLisandro Dalcin 
225f996e368SLisandro Dalcin   PetscFunctionBegin;
226f996e368SLisandro Dalcin   ierr = MPI_Comm_rank(comm, &rank);CHKERRMPI(ierr);
227f996e368SLisandro Dalcin   if (!rank) {
228f996e368SLisandro Dalcin     char   fpath[PETSC_MAX_PATH_LEN];
229f996e368SLisandro Dalcin     char   fname[PETSC_MAX_PATH_LEN];
230f996e368SLisandro Dalcin     FILE  *fd;
231f996e368SLisandro Dalcin     size_t rd;
232f996e368SLisandro Dalcin 
233f996e368SLisandro Dalcin     ierr = PetscStrreplace(PETSC_COMM_SELF, file, fpath, sizeof(fpath));CHKERRQ(ierr);
234f996e368SLisandro Dalcin     ierr = PetscFixFilename(fpath, fname);CHKERRQ(ierr);
235f996e368SLisandro Dalcin 
236f996e368SLisandro Dalcin     fd = fopen(fname, "r");
237f996e368SLisandro Dalcin     if (fd) {
238f996e368SLisandro Dalcin       fseek(fd, 0, SEEK_END);
239f996e368SLisandro Dalcin       yamlLength = (int)ftell(fd);
240f996e368SLisandro Dalcin       fseek(fd, 0, SEEK_SET);
241f996e368SLisandro Dalcin       if (yamlLength < 0) SETERRQ1(PETSC_COMM_SELF, PETSC_ERR_FILE_OPEN, "Unable to query size of YAML file: %s", fname);
242f996e368SLisandro Dalcin       ierr = PetscMalloc1(yamlLength+1, &yamlString);CHKERRQ(ierr);
243f996e368SLisandro Dalcin       rd = fread(yamlString, 1, (size_t)yamlLength, fd);
244f996e368SLisandro Dalcin       if (rd != (size_t)yamlLength) SETERRQ1(PETSC_COMM_SELF, PETSC_ERR_FILE_READ, "Unable to read entire YAML file: %s", fname);
245f996e368SLisandro Dalcin       yamlString[yamlLength] = 0;
246f996e368SLisandro Dalcin       fclose(fd);
247f996e368SLisandro Dalcin     }
248f996e368SLisandro Dalcin   }
249f996e368SLisandro Dalcin 
250f996e368SLisandro Dalcin   ierr = MPI_Bcast(&yamlLength, 1, MPI_INT, 0, comm);CHKERRMPI(ierr);
251f996e368SLisandro Dalcin   if (require && yamlLength < 0) SETERRQ1(comm, PETSC_ERR_FILE_OPEN, "Unable to open YAML option file: %s\n", file);
252f996e368SLisandro Dalcin   if (yamlLength < 0) PetscFunctionReturn(0);
253f996e368SLisandro Dalcin 
254f996e368SLisandro Dalcin   if (rank) {ierr = PetscMalloc1(yamlLength+1, &yamlString);CHKERRQ(ierr);}
255f996e368SLisandro Dalcin   ierr = MPI_Bcast(yamlString, yamlLength+1, MPI_CHAR, 0, comm);CHKERRMPI(ierr);
256f996e368SLisandro Dalcin 
257f996e368SLisandro Dalcin   prev = PetscYAMLSetComm(comm);
258f996e368SLisandro Dalcin   ierr = PetscOptionsInsertStringYAML(options, yamlString);CHKERRQ(ierr);
259f996e368SLisandro Dalcin   (void) PetscYAMLSetComm(prev);
260f996e368SLisandro Dalcin 
261f996e368SLisandro Dalcin   ierr = PetscFree(yamlString);CHKERRQ(ierr);
262f996e368SLisandro Dalcin   PetscFunctionReturn(0);
263f996e368SLisandro Dalcin }
264951eb098SLisandro Dalcin 
265951eb098SLisandro Dalcin 
266951eb098SLisandro Dalcin #if !defined(PETSC_HAVE_YAML)
267951eb098SLisandro Dalcin 
268951eb098SLisandro Dalcin /*
269951eb098SLisandro Dalcin #if !defined(PETSC_HAVE_STRDUP)
270951eb098SLisandro Dalcin #define strdup(s) (char*)memcpy(malloc(strlen(s)+1),s,strlen(s)+1)
271951eb098SLisandro Dalcin #endif
272951eb098SLisandro Dalcin */
273951eb098SLisandro Dalcin 
274951eb098SLisandro Dalcin /* Embed LibYAML in this compilation unit */
275951eb098SLisandro Dalcin #include <../src/sys/yaml/src/api.c>
276951eb098SLisandro Dalcin #include <../src/sys/yaml/src/loader.c>
277951eb098SLisandro Dalcin #include <../src/sys/yaml/src/parser.c>
278951eb098SLisandro Dalcin #include <../src/sys/yaml/src/reader.c>
279951eb098SLisandro Dalcin #include <../src/sys/yaml/src/scanner.c>
280951eb098SLisandro Dalcin 
281951eb098SLisandro Dalcin /* Silence a few unused-function warnings */
282951eb098SLisandro Dalcin static PETSC_UNUSED void petsc_yaml_unused(void)
283951eb098SLisandro Dalcin {
284951eb098SLisandro Dalcin   (void)yaml_parser_scan;
285951eb098SLisandro Dalcin   (void)yaml_document_get_node;
286951eb098SLisandro Dalcin   (void)yaml_parser_set_encoding;
287951eb098SLisandro Dalcin   (void)yaml_parser_set_input;
288951eb098SLisandro Dalcin   (void)yaml_parser_set_input_file;
289951eb098SLisandro Dalcin }
290951eb098SLisandro Dalcin 
291951eb098SLisandro Dalcin #endif
292