blob: e56a25255157f93e634d26c41dd253d0a1b2f11a [file] [log] [blame]
Radek Krejcid3ca0632019-04-16 16:54:54 +02001/**
2 * @file printer.c
3 * @author Radek Krejci <rkrejci@cesnet.cz>
Radek Krejcie7b95092019-05-15 11:03:07 +02004 * @brief Generic libyang printers functions.
Radek Krejcid3ca0632019-04-16 16:54:54 +02005 *
6 * Copyright (c) 2015 - 2019 CESNET, z.s.p.o.
7 *
8 * This source code is licensed under BSD 3-Clause License (the "License").
9 * You may not use this file except in compliance with the License.
10 * You may obtain a copy of the License at
11 *
12 * https://opensource.org/licenses/BSD-3-Clause
13 */
14
Radek Krejci535ea9f2020-05-29 16:01:05 +020015#define _GNU_SOURCE
Radek Krejcid3ca0632019-04-16 16:54:54 +020016
Radek Krejci535ea9f2020-05-29 16:01:05 +020017#include "printer.h"
18
19#include <assert.h>
Radek Krejcid3ca0632019-04-16 16:54:54 +020020#include <errno.h>
21#include <stdarg.h>
Radek Krejcie7b95092019-05-15 11:03:07 +020022#include <stdio.h>
23#include <stdlib.h>
Radek Krejcid3ca0632019-04-16 16:54:54 +020024#include <string.h>
Radek Krejcie7b95092019-05-15 11:03:07 +020025#include <unistd.h>
Radek Krejcid3ca0632019-04-16 16:54:54 +020026
Radek Krejci535ea9f2020-05-29 16:01:05 +020027#include "common.h"
28#include "config.h"
Radek Krejcie7b95092019-05-15 11:03:07 +020029#include "log.h"
Michal Vasko9b368d32020-02-14 13:53:31 +010030#include "plugins_types.h"
Radek Krejci535ea9f2020-05-29 16:01:05 +020031#include "printer_data.h"
32#include "printer_internal.h"
33#include "tree.h"
34#include "tree_schema.h"
Radek Krejcid3ca0632019-04-16 16:54:54 +020035
36/**
37 * @brief informational structure shared by printers
38 */
39struct ext_substmt_info_s ext_substmt_info[] = {
40 {NULL, NULL, 0}, /**< LYEXT_SUBSTMT_SELF */
41 {"argument", "name", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_ARGUMENT */
42 {"base", "name", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_BASE */
43 {"belongs-to", "module", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_BELONGSTO */
44 {"contact", "text", SUBST_FLAG_YIN}, /**< LYEXT_SUBSTMT_CONTACT */
45 {"default", "value", 0}, /**< LYEXT_SUBSTMT_DEFAULT */
46 {"description", "text", SUBST_FLAG_YIN}, /**< LYEXT_SUBSTMT_DESCRIPTION */
47 {"error-app-tag", "value", 0}, /**< LYEXT_SUBSTMT_ERRTAG */
48 {"error-message", "value", SUBST_FLAG_YIN}, /**< LYEXT_SUBSTMT_ERRMSG */
49 {"key", "value", 0}, /**< LYEXT_SUBSTMT_KEY */
50 {"namespace", "uri", 0}, /**< LYEXT_SUBSTMT_NAMESPACE */
51 {"organization", "text", SUBST_FLAG_YIN}, /**< LYEXT_SUBSTMT_ORGANIZATION */
52 {"path", "value", 0}, /**< LYEXT_SUBSTMT_PATH */
53 {"prefix", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_PREFIX */
54 {"presence", "value", 0}, /**< LYEXT_SUBSTMT_PRESENCE */
55 {"reference", "text", SUBST_FLAG_YIN}, /**< LYEXT_SUBSTMT_REFERENCE */
56 {"revision-date", "date", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_REVISIONDATE */
57 {"units", "name", 0}, /**< LYEXT_SUBSTMT_UNITS */
58 {"value", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_VALUE */
59 {"yang-version", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_VERSION */
60 {"modifier", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_MODIFIER */
61 {"require-instance", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_REQINST */
62 {"yin-element", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_YINELEM */
63 {"config", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_CONFIG */
64 {"mandatory", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_MANDATORY */
65 {"ordered-by", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_ORDEREDBY */
66 {"status", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_STATUS */
67 {"fraction-digits", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_DIGITS */
68 {"max-elements", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_MAX */
69 {"min-elements", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_MIN */
70 {"position", "value", SUBST_FLAG_ID}, /**< LYEXT_SUBSTMT_POSITION */
71 {"unique", "tag", 0}, /**< LYEXT_SUBSTMT_UNIQUE */
72};
73
Michal Vasko6f4cbb62020-02-28 11:15:47 +010074int
Michal Vasko9b368d32020-02-14 13:53:31 +010075ly_is_default(const struct lyd_node *node)
76{
77 const struct lysc_node_leaf *leaf;
78 const struct lysc_node_leaflist *llist;
79 const struct lyd_node_term *term;
Radek Krejci7eb54ba2020-05-18 16:30:04 +020080 LY_ARRAY_SIZE_TYPE u;
Michal Vasko9b368d32020-02-14 13:53:31 +010081
82 assert(node->schema->nodetype & LYD_NODE_TERM);
83 term = (const struct lyd_node_term *)node;
84
85 if (node->schema->nodetype == LYS_LEAF) {
86 leaf = (const struct lysc_node_leaf *)node->schema;
87 if (!leaf->dflt) {
88 return 0;
89 }
90
91 /* compare with the default value */
92 if (leaf->type->plugin->compare(&term->value, leaf->dflt)) {
93 return 0;
94 }
95 } else {
96 llist = (const struct lysc_node_leaflist *)node->schema;
97 if (!llist->dflts) {
98 return 0;
99 }
100
Radek Krejci7eb54ba2020-05-18 16:30:04 +0200101 LY_ARRAY_FOR(llist->dflts, u) {
Michal Vasko9b368d32020-02-14 13:53:31 +0100102 /* compare with each possible default value */
Radek Krejci7eb54ba2020-05-18 16:30:04 +0200103 if (llist->type->plugin->compare(&term->value, llist->dflts[u])) {
Michal Vasko9b368d32020-02-14 13:53:31 +0100104 return 0;
105 }
106 }
107 }
108
109 return 1;
110}
111
112int
113ly_should_print(const struct lyd_node *node, int options)
114{
115 const struct lyd_node *next, *elem;
116
117 if (options & LYDP_WD_TRIM) {
118 /* do not print default nodes */
119 if (node->flags & LYD_DEFAULT) {
120 /* implicit default node/NP container with only default nodes */
121 return 0;
122 } else if (node->schema->nodetype & LYD_NODE_TERM) {
123 if (ly_is_default(node)) {
124 /* explicit default node */
125 return 0;
126 }
127 }
128 } else if ((node->flags & LYD_DEFAULT) && !(options & LYDP_WD_MASK) && !(node->schema->flags & LYS_CONFIG_R)) {
Radek Krejci241f6b52020-05-21 18:13:49 +0200129 /* LYDP_WD_EXPLICIT
Michal Vasko9b368d32020-02-14 13:53:31 +0100130 * - print only if it contains status data in its subtree */
131 LYD_TREE_DFS_BEGIN(node, next, elem) {
Michal Vaskodb4f9e42020-06-01 17:29:56 +0200132 if ((elem->schema->nodetype != LYS_CONTAINER) || (elem->schema->flags & LYS_PRESENCE)) {
133 if (elem->schema->flags & LYS_CONFIG_R) {
134 return 1;
135 }
Michal Vasko9b368d32020-02-14 13:53:31 +0100136 }
137 LYD_TREE_DFS_END(node, next, elem)
138 }
139 return 0;
140 } else if ((node->flags & LYD_DEFAULT) && (node->schema->nodetype == LYS_CONTAINER) && !(options & LYDP_KEEPEMPTYCONT)) {
141 /* avoid empty default containers */
142 LYD_TREE_DFS_BEGIN(node, next, elem) {
143 if (elem->schema->nodetype != LYS_CONTAINER) {
144 return 1;
145 }
146 assert(elem->flags & LYD_DEFAULT);
147 LYD_TREE_DFS_END(node, next, elem)
148 }
149 return 0;
150 }
151
152 return 1;
153}
154
Radek Krejci241f6b52020-05-21 18:13:49 +0200155API LY_OUT_TYPE
156ly_out_type(const struct ly_out *out)
Radek Krejcia5bba312020-01-09 15:41:20 +0100157{
Radek Krejci241f6b52020-05-21 18:13:49 +0200158 LY_CHECK_ARG_RET(NULL, out, LY_OUT_ERROR);
Radek Krejcia5bba312020-01-09 15:41:20 +0100159 return out->type;
160}
161
Radek Krejci84ce7b12020-06-11 17:28:25 +0200162API LY_ERR
163ly_out_new_clb(ssize_t (*writeclb)(void *arg, const void *buf, size_t count), void *arg, struct ly_out **out)
Radek Krejcia5bba312020-01-09 15:41:20 +0100164{
Radek Krejci84ce7b12020-06-11 17:28:25 +0200165 LY_CHECK_ARG_RET(NULL, out, writeclb, LY_EINVAL);
Radek Krejcia5bba312020-01-09 15:41:20 +0100166
Radek Krejci84ce7b12020-06-11 17:28:25 +0200167 *out = calloc(1, sizeof **out);
168 LY_CHECK_ERR_RET(!*out, LOGMEM(NULL), LY_EMEM);
Radek Krejcia5bba312020-01-09 15:41:20 +0100169
Radek Krejci84ce7b12020-06-11 17:28:25 +0200170 (*out)->type = LY_OUT_CALLBACK;
171 (*out)->method.clb.func = writeclb;
172 (*out)->method.clb.arg = arg;
Radek Krejcia5bba312020-01-09 15:41:20 +0100173
Radek Krejci84ce7b12020-06-11 17:28:25 +0200174 return LY_SUCCESS;
Radek Krejcia5bba312020-01-09 15:41:20 +0100175}
176
Radek Krejci241f6b52020-05-21 18:13:49 +0200177API ssize_t (*ly_out_clb(struct ly_out *out, ssize_t (*writeclb)(void *arg, const void *buf, size_t count)))(void *arg, const void *buf, size_t count)
Radek Krejcia5bba312020-01-09 15:41:20 +0100178{
179 void *prev_clb;
180
Radek Krejci241f6b52020-05-21 18:13:49 +0200181 LY_CHECK_ARG_RET(NULL, out, out->type == LY_OUT_CALLBACK, NULL);
Radek Krejcia5bba312020-01-09 15:41:20 +0100182
183 prev_clb = out->method.clb.func;
184
185 if (writeclb) {
186 out->method.clb.func = writeclb;
187 }
188
189 return prev_clb;
190}
191
192API void *
Radek Krejci241f6b52020-05-21 18:13:49 +0200193ly_out_clb_arg(struct ly_out *out, void *arg)
Radek Krejcia5bba312020-01-09 15:41:20 +0100194{
195 void *prev_arg;
196
Radek Krejci241f6b52020-05-21 18:13:49 +0200197 LY_CHECK_ARG_RET(NULL, out, out->type == LY_OUT_CALLBACK, NULL);
Radek Krejcia5bba312020-01-09 15:41:20 +0100198
199 prev_arg = out->method.clb.arg;
200
201 if (arg) {
202 out->method.clb.arg = arg;
203 }
204
205 return prev_arg;
206}
207
Radek Krejci84ce7b12020-06-11 17:28:25 +0200208API LY_ERR
209ly_out_new_fd(int fd, struct ly_out **out)
Radek Krejcia5bba312020-01-09 15:41:20 +0100210{
Radek Krejci84ce7b12020-06-11 17:28:25 +0200211 LY_CHECK_ARG_RET(NULL, out, fd != -1, LY_EINVAL);
Radek Krejcia5bba312020-01-09 15:41:20 +0100212
Radek Krejci84ce7b12020-06-11 17:28:25 +0200213 *out = calloc(1, sizeof **out);
214 LY_CHECK_ERR_RET(!*out, LOGMEM(NULL), LY_EMEM);
Radek Krejcia5bba312020-01-09 15:41:20 +0100215
216#ifdef HAVE_VDPRINTF
Radek Krejci84ce7b12020-06-11 17:28:25 +0200217 (*out)->type = LY_OUT_FD;
218 (*out)->method.fd = fd;
Radek Krejcia5bba312020-01-09 15:41:20 +0100219#else
220 /* Without vdfprintf(), change the printing method to printing to a FILE stream.
221 * To preserve the original file descriptor, duplicate it and use it to open file stream. */
Radek Krejci84ce7b12020-06-11 17:28:25 +0200222 (*out)->type = LY_OUT_FDSTREAM;
223 (*out)->method.fdstream.fd = fd;
Radek Krejcia5bba312020-01-09 15:41:20 +0100224
Radek Krejci84ce7b12020-06-11 17:28:25 +0200225 fd = dup((*out)->method.fdstream.fd);
Radek Krejcia5bba312020-01-09 15:41:20 +0100226 if (fd < 0) {
227 LOGERR(NULL, LY_ESYS, "Unable to duplicate provided file descriptor (%d) for printing the output (%s).",
Radek Krejci84ce7b12020-06-11 17:28:25 +0200228 (*out)->method.fdstream.fd, strerror(errno));
229 free(*out);
230 *out = NULL;
231 return LY_ESYS;
Radek Krejcia5bba312020-01-09 15:41:20 +0100232 }
Radek Krejci84ce7b12020-06-11 17:28:25 +0200233 (*out)->method.fdstream.f = fdopen(fd, "a");
234 if (!(*out)->method.fdstream.f) {
Radek Krejcia5bba312020-01-09 15:41:20 +0100235 LOGERR(NULL, LY_ESYS, "Unable to open provided file descriptor (%d) for printing the output (%s).",
Radek Krejci84ce7b12020-06-11 17:28:25 +0200236 (*out)->method.fdstream.fd, strerror(errno));
237 free(*out);
238 *out = NULL;
Radek Krejci7d1e62a2020-06-29 17:34:08 +0200239 close(fd);
Radek Krejci84ce7b12020-06-11 17:28:25 +0200240 return LY_ESYS;
Radek Krejcia5bba312020-01-09 15:41:20 +0100241 }
242#endif
243
Radek Krejci84ce7b12020-06-11 17:28:25 +0200244 return LY_SUCCESS;
Radek Krejcia5bba312020-01-09 15:41:20 +0100245}
246
247API int
Radek Krejci241f6b52020-05-21 18:13:49 +0200248ly_out_fd(struct ly_out *out, int fd)
Radek Krejcia5bba312020-01-09 15:41:20 +0100249{
250 int prev_fd;
251
Radek Krejci241f6b52020-05-21 18:13:49 +0200252 LY_CHECK_ARG_RET(NULL, out, out->type <= LY_OUT_FDSTREAM, -1);
Radek Krejcia5bba312020-01-09 15:41:20 +0100253
Radek Krejci241f6b52020-05-21 18:13:49 +0200254 if (out->type == LY_OUT_FDSTREAM) {
Radek Krejcia5bba312020-01-09 15:41:20 +0100255 prev_fd = out->method.fdstream.fd;
Radek Krejci241f6b52020-05-21 18:13:49 +0200256 } else { /* LY_OUT_FD */
Radek Krejcia5bba312020-01-09 15:41:20 +0100257 prev_fd = out->method.fd;
258 }
259
260 if (fd != -1) {
261 /* replace output stream */
Radek Krejci241f6b52020-05-21 18:13:49 +0200262 if (out->type == LY_OUT_FDSTREAM) {
Radek Krejcia5bba312020-01-09 15:41:20 +0100263 int streamfd;
264 FILE *stream;
265
266 streamfd = dup(fd);
267 if (streamfd < 0) {
268 LOGERR(NULL, LY_ESYS, "Unable to duplicate provided file descriptor (%d) for printing the output (%s).", fd, strerror(errno));
269 return -1;
270 }
271 stream = fdopen(streamfd, "a");
272 if (!stream) {
273 LOGERR(NULL, LY_ESYS, "Unable to open provided file descriptor (%d) for printing the output (%s).", fd, strerror(errno));
274 close(streamfd);
275 return -1;
276 }
277 /* close only the internally created stream, file descriptor is returned and supposed to be closed by the caller */
278 fclose(out->method.fdstream.f);
279 out->method.fdstream.f = stream;
280 out->method.fdstream.fd = streamfd;
Radek Krejci241f6b52020-05-21 18:13:49 +0200281 } else { /* LY_OUT_FD */
Radek Krejcia5bba312020-01-09 15:41:20 +0100282 out->method.fd = fd;
283 }
284 }
285
286 return prev_fd;
287}
288
Radek Krejci84ce7b12020-06-11 17:28:25 +0200289API LY_ERR
290ly_out_new_file(FILE *f, struct ly_out **out)
Radek Krejcia5bba312020-01-09 15:41:20 +0100291{
Radek Krejci84ce7b12020-06-11 17:28:25 +0200292 LY_CHECK_ARG_RET(NULL, out, f, LY_EINVAL);
Radek Krejcia5bba312020-01-09 15:41:20 +0100293
Radek Krejci84ce7b12020-06-11 17:28:25 +0200294 *out = calloc(1, sizeof **out);
295 LY_CHECK_ERR_RET(!*out, LOGMEM(NULL), LY_EMEM);
Radek Krejcia5bba312020-01-09 15:41:20 +0100296
Radek Krejci84ce7b12020-06-11 17:28:25 +0200297 (*out)->type = LY_OUT_FILE;
298 (*out)->method.f = f;
Radek Krejcia5bba312020-01-09 15:41:20 +0100299
Radek Krejci84ce7b12020-06-11 17:28:25 +0200300 return LY_SUCCESS;
Radek Krejcia5bba312020-01-09 15:41:20 +0100301}
302
303API FILE *
Radek Krejci241f6b52020-05-21 18:13:49 +0200304ly_out_file(struct ly_out *out, FILE *f)
Radek Krejcia5bba312020-01-09 15:41:20 +0100305{
306 FILE *prev_f;
307
Radek Krejci241f6b52020-05-21 18:13:49 +0200308 LY_CHECK_ARG_RET(NULL, out, out->type == LY_OUT_FILE, NULL);
Radek Krejcia5bba312020-01-09 15:41:20 +0100309
310 prev_f = out->method.f;
311
312 if (f) {
313 out->method.f = f;
314 }
315
316 return prev_f;
317}
318
Radek Krejci84ce7b12020-06-11 17:28:25 +0200319API LY_ERR
320ly_out_new_memory(char **strp, size_t size, struct ly_out **out)
Radek Krejcia5bba312020-01-09 15:41:20 +0100321{
Radek Krejci84ce7b12020-06-11 17:28:25 +0200322 LY_CHECK_ARG_RET(NULL, out, strp, LY_EINVAL);
Radek Krejcia5bba312020-01-09 15:41:20 +0100323
Radek Krejci84ce7b12020-06-11 17:28:25 +0200324 *out = calloc(1, sizeof **out);
325 LY_CHECK_ERR_RET(!*out, LOGMEM(NULL), LY_EMEM);
Radek Krejcia5bba312020-01-09 15:41:20 +0100326
Radek Krejci84ce7b12020-06-11 17:28:25 +0200327 (*out)->type = LY_OUT_MEMORY;
328 (*out)->method.mem.buf = strp;
Radek Krejcia5bba312020-01-09 15:41:20 +0100329 if (!size) {
330 /* buffer is supposed to be allocated */
331 *strp = NULL;
332 } else if (*strp) {
333 /* there is already buffer to use */
Radek Krejci84ce7b12020-06-11 17:28:25 +0200334 (*out)->method.mem.size = size;
Radek Krejcia5bba312020-01-09 15:41:20 +0100335 }
336
Radek Krejci84ce7b12020-06-11 17:28:25 +0200337 return LY_SUCCESS;
Radek Krejcia5bba312020-01-09 15:41:20 +0100338}
339
340char *
Radek Krejci241f6b52020-05-21 18:13:49 +0200341ly_out_memory(struct ly_out *out, char **strp, size_t size)
Radek Krejcia5bba312020-01-09 15:41:20 +0100342{
343 char *data;
344
Radek Krejci241f6b52020-05-21 18:13:49 +0200345 LY_CHECK_ARG_RET(NULL, out, out->type == LY_OUT_MEMORY, NULL);
Radek Krejcia5bba312020-01-09 15:41:20 +0100346
347 data = *out->method.mem.buf;
348
349 if (strp) {
350 out->method.mem.buf = strp;
351 out->method.mem.len = out->method.mem.size = 0;
352 out->printed = 0;
353 if (!size) {
354 /* buffer is supposed to be allocated */
355 *strp = NULL;
356 } else if (*strp) {
357 /* there is already buffer to use */
358 out->method.mem.size = size;
359 }
360 }
361
362 return data;
363}
364
365API LY_ERR
Radek Krejci241f6b52020-05-21 18:13:49 +0200366ly_out_reset(struct ly_out *out)
Radek Krejcia5bba312020-01-09 15:41:20 +0100367{
368 LY_CHECK_ARG_RET(NULL, out, LY_EINVAL);
369
370 switch(out->type) {
Radek Krejci241f6b52020-05-21 18:13:49 +0200371 case LY_OUT_ERROR:
Radek Krejcia5bba312020-01-09 15:41:20 +0100372 LOGINT(NULL);
373 return LY_EINT;
Radek Krejci241f6b52020-05-21 18:13:49 +0200374 case LY_OUT_FD:
Radek Krejcia5bba312020-01-09 15:41:20 +0100375 if ((lseek(out->method.fd, 0, SEEK_SET) == -1) && errno != ESPIPE) {
376 LOGERR(NULL, LY_ESYS, "Seeking output file descriptor failed (%s).", strerror(errno));
377 return LY_ESYS;
378 }
Radek Krejcic5a12e12020-05-27 17:09:59 +0200379 if (errno != ESPIPE && ftruncate(out->method.fd, 0) == -1) {
380 LOGERR(NULL, LY_ESYS, "Truncating output file failed (%s).", strerror(errno));
381 return LY_ESYS;
382 }
Radek Krejcia5bba312020-01-09 15:41:20 +0100383 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200384 case LY_OUT_FDSTREAM:
385 case LY_OUT_FILE:
386 case LY_OUT_FILEPATH:
Radek Krejcia5bba312020-01-09 15:41:20 +0100387 if ((fseek(out->method.f, 0, SEEK_SET) == -1) && errno != ESPIPE) {
388 LOGERR(NULL, LY_ESYS, "Seeking output file stream failed (%s).", strerror(errno));
389 return LY_ESYS;
390 }
Radek Krejcic5a12e12020-05-27 17:09:59 +0200391 if (errno != ESPIPE && ftruncate(fileno(out->method.f), 0) == -1) {
392 LOGERR(NULL, LY_ESYS, "Truncating output file failed (%s).", strerror(errno));
393 return LY_ESYS;
394 }
Radek Krejcia5bba312020-01-09 15:41:20 +0100395 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200396 case LY_OUT_MEMORY:
Radek Krejcic5a12e12020-05-27 17:09:59 +0200397 if (out->method.mem.buf && *out->method.mem.buf) {
398 memset(*out->method.mem.buf, 0, out->method.mem.len);
399 }
Radek Krejcia5bba312020-01-09 15:41:20 +0100400 out->printed = 0;
401 out->method.mem.len = 0;
402 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200403 case LY_OUT_CALLBACK:
Radek Krejcia5bba312020-01-09 15:41:20 +0100404 /* nothing to do (not seekable) */
405 break;
406 }
407
408 return LY_SUCCESS;
409}
410
Radek Krejci84ce7b12020-06-11 17:28:25 +0200411API LY_ERR
412ly_out_new_filepath(const char *filepath, struct ly_out **out)
Radek Krejcia5bba312020-01-09 15:41:20 +0100413{
Radek Krejci84ce7b12020-06-11 17:28:25 +0200414 LY_CHECK_ARG_RET(NULL, out, filepath, LY_EINVAL);
Radek Krejcia5bba312020-01-09 15:41:20 +0100415
Radek Krejci84ce7b12020-06-11 17:28:25 +0200416 *out = calloc(1, sizeof **out);
417 LY_CHECK_ERR_RET(!*out, LOGMEM(NULL), LY_EMEM);
Radek Krejcia5bba312020-01-09 15:41:20 +0100418
Radek Krejci84ce7b12020-06-11 17:28:25 +0200419 (*out)->type = LY_OUT_FILEPATH;
420 (*out)->method.fpath.f = fopen(filepath, "w");
421 if (!(*out)->method.fpath.f) {
Radek Krejcia5bba312020-01-09 15:41:20 +0100422 LOGERR(NULL, LY_ESYS, "Failed to open file \"%s\" (%s).", filepath, strerror(errno));
Radek Krejci84ce7b12020-06-11 17:28:25 +0200423 return LY_ESYS;
Radek Krejcia5bba312020-01-09 15:41:20 +0100424 }
Radek Krejci84ce7b12020-06-11 17:28:25 +0200425 (*out)->method.fpath.filepath = strdup(filepath);
426 return LY_SUCCESS;
Radek Krejcia5bba312020-01-09 15:41:20 +0100427}
428
429API const char *
Radek Krejci241f6b52020-05-21 18:13:49 +0200430ly_out_filepath(struct ly_out *out, const char *filepath)
Radek Krejcia5bba312020-01-09 15:41:20 +0100431{
432 FILE *f;
433
Radek Krejci241f6b52020-05-21 18:13:49 +0200434 LY_CHECK_ARG_RET(NULL, out, out->type == LY_OUT_FILEPATH, filepath ? NULL : ((void *)-1));
Radek Krejcia5bba312020-01-09 15:41:20 +0100435
436 if (!filepath) {
437 return out->method.fpath.filepath;
438 }
439
440 /* replace filepath */
441 f = out->method.fpath.f;
442 out->method.fpath.f = fopen(filepath, "w");
443 if (!out->method.fpath.f) {
444 LOGERR(NULL, LY_ESYS, "Failed to open file \"%s\" (%s).", filepath, strerror(errno));
445 out->method.fpath.f = f;
446 return ((void *)-1);
447 }
448 fclose(f);
449 free(out->method.fpath.filepath);
450 out->method.fpath.filepath = strdup(filepath);
451
452 return NULL;
453}
454
455API void
Radek Krejci241f6b52020-05-21 18:13:49 +0200456ly_out_free(struct ly_out *out, void (*clb_arg_destructor)(void *arg), int destroy)
Radek Krejcia5bba312020-01-09 15:41:20 +0100457{
458 if (!out) {
459 return;
460 }
461
462 switch (out->type) {
Radek Krejci241f6b52020-05-21 18:13:49 +0200463 case LY_OUT_CALLBACK:
Radek Krejcia5bba312020-01-09 15:41:20 +0100464 if (clb_arg_destructor) {
465 clb_arg_destructor(out->method.clb.arg);
466 }
467 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200468 case LY_OUT_FDSTREAM:
Radek Krejcia5bba312020-01-09 15:41:20 +0100469 fclose(out->method.fdstream.f);
470 if (destroy) {
471 close(out->method.fdstream.fd);
472 }
473 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200474 case LY_OUT_FD:
Radek Krejcia5bba312020-01-09 15:41:20 +0100475 if (destroy) {
476 close(out->method.fd);
477 }
478 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200479 case LY_OUT_FILE:
Radek Krejcia5bba312020-01-09 15:41:20 +0100480 if (destroy) {
481 fclose(out->method.f);
482 }
483 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200484 case LY_OUT_MEMORY:
Radek Krejcia5bba312020-01-09 15:41:20 +0100485 if (destroy) {
486 free(*out->method.mem.buf);
487 }
488 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200489 case LY_OUT_FILEPATH:
Radek Krejcia5bba312020-01-09 15:41:20 +0100490 free(out->method.fpath.filepath);
Radek Krejci2aae3752020-05-27 18:16:30 +0200491 fclose(out->method.fpath.f);
Radek Krejcia5bba312020-01-09 15:41:20 +0100492 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200493 case LY_OUT_ERROR:
Radek Krejcia5bba312020-01-09 15:41:20 +0100494 LOGINT(NULL);
495 }
496 free(out);
497}
498
Radek Krejcibaeb8382020-05-27 16:44:53 +0200499API ssize_t
Radek Krejci241f6b52020-05-21 18:13:49 +0200500ly_print(struct ly_out *out, const char *format, ...)
Radek Krejcid3ca0632019-04-16 16:54:54 +0200501{
502 int count = 0;
503 char *msg = NULL, *aux;
504 va_list ap;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200505
Radek Krejcibaeb8382020-05-27 16:44:53 +0200506 LYOUT_CHECK(out, -1 * out->status);
Radek Krejci56cc0872019-04-30 09:22:27 +0200507
Radek Krejcid3ca0632019-04-16 16:54:54 +0200508 va_start(ap, format);
509
510 switch (out->type) {
Radek Krejci241f6b52020-05-21 18:13:49 +0200511 case LY_OUT_FD:
Radek Krejcid3ca0632019-04-16 16:54:54 +0200512#ifdef HAVE_VDPRINTF
513 count = vdprintf(out->method.fd, format, ap);
Radek Krejcid3ca0632019-04-16 16:54:54 +0200514 break;
Radek Krejci4a0ed4a2019-04-18 15:08:34 +0200515#else
Radek Krejci241f6b52020-05-21 18:13:49 +0200516 /* never should be here since ly_out_fd() is supposed to set type to LY_OUT_FDSTREAM in case vdprintf() is missing */
Radek Krejcia5bba312020-01-09 15:41:20 +0100517 LOGINT(NULL);
Radek Krejcibaeb8382020-05-27 16:44:53 +0200518 va_end(ap);
519 return -LY_EINT;
Radek Krejci4a0ed4a2019-04-18 15:08:34 +0200520#endif
Radek Krejci241f6b52020-05-21 18:13:49 +0200521 case LY_OUT_FDSTREAM:
522 case LY_OUT_FILEPATH:
523 case LY_OUT_FILE:
Radek Krejcid3ca0632019-04-16 16:54:54 +0200524 count = vfprintf(out->method.f, format, ap);
525 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200526 case LY_OUT_MEMORY:
Radek Krejci897ad2e2019-04-29 16:43:07 +0200527 if ((count = vasprintf(&msg, format, ap)) < 0) {
528 break;
529 }
Radek Krejcid3ca0632019-04-16 16:54:54 +0200530 if (out->method.mem.len + count + 1 > out->method.mem.size) {
Radek Krejcia5bba312020-01-09 15:41:20 +0100531 aux = ly_realloc(*out->method.mem.buf, out->method.mem.len + count + 1);
Radek Krejcid3ca0632019-04-16 16:54:54 +0200532 if (!aux) {
533 out->method.mem.buf = NULL;
534 out->method.mem.len = 0;
535 out->method.mem.size = 0;
536 LOGMEM(NULL);
537 va_end(ap);
Radek Krejcibaeb8382020-05-27 16:44:53 +0200538 return -LY_EMEM;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200539 }
Radek Krejcia5bba312020-01-09 15:41:20 +0100540 *out->method.mem.buf = aux;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200541 out->method.mem.size = out->method.mem.len + count + 1;
542 }
Radek Krejcia5bba312020-01-09 15:41:20 +0100543 memcpy(&(*out->method.mem.buf)[out->method.mem.len], msg, count);
Radek Krejcid3ca0632019-04-16 16:54:54 +0200544 out->method.mem.len += count;
Radek Krejcia5bba312020-01-09 15:41:20 +0100545 (*out->method.mem.buf)[out->method.mem.len] = '\0';
Radek Krejcid3ca0632019-04-16 16:54:54 +0200546 free(msg);
547 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200548 case LY_OUT_CALLBACK:
Radek Krejci897ad2e2019-04-29 16:43:07 +0200549 if ((count = vasprintf(&msg, format, ap)) < 0) {
550 break;
551 }
Radek Krejcia5bba312020-01-09 15:41:20 +0100552 count = out->method.clb.func(out->method.clb.arg, msg, count);
Radek Krejcid3ca0632019-04-16 16:54:54 +0200553 free(msg);
554 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200555 case LY_OUT_ERROR:
Radek Krejcia5bba312020-01-09 15:41:20 +0100556 LOGINT(NULL);
Radek Krejcibaeb8382020-05-27 16:44:53 +0200557 va_end(ap);
558 return -LY_EINT;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200559 }
560
561 va_end(ap);
562
563 if (count < 0) {
Radek Krejci897ad2e2019-04-29 16:43:07 +0200564 LOGERR(out->ctx, LY_ESYS, "%s: writing data failed (%s).", __func__, strerror(errno));
Radek Krejci56cc0872019-04-30 09:22:27 +0200565 out->status = LY_ESYS;
Radek Krejcibaeb8382020-05-27 16:44:53 +0200566 return -LY_ESYS;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200567 } else {
Radek Krejci241f6b52020-05-21 18:13:49 +0200568 if (out->type == LY_OUT_FDSTREAM) {
Radek Krejcia5bba312020-01-09 15:41:20 +0100569 /* move the original file descriptor to the end of the output file */
570 lseek(out->method.fdstream.fd, 0, SEEK_END);
571 }
Radek Krejci897ad2e2019-04-29 16:43:07 +0200572 out->printed += count;
Radek Krejcibaeb8382020-05-27 16:44:53 +0200573 return count;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200574 }
575}
576
Radek Krejci2aae3752020-05-27 18:16:30 +0200577API void
Radek Krejci241f6b52020-05-21 18:13:49 +0200578ly_print_flush(struct ly_out *out)
Radek Krejcid3ca0632019-04-16 16:54:54 +0200579{
580 switch (out->type) {
Radek Krejci241f6b52020-05-21 18:13:49 +0200581 case LY_OUT_FDSTREAM:
Radek Krejcia5bba312020-01-09 15:41:20 +0100582 /* move the original file descriptor to the end of the output file */
583 lseek(out->method.fdstream.fd, 0, SEEK_END);
584 fflush(out->method.fdstream.f);
585 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200586 case LY_OUT_FILEPATH:
587 case LY_OUT_FILE:
Radek Krejcid3ca0632019-04-16 16:54:54 +0200588 fflush(out->method.f);
589 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200590 case LY_OUT_FD:
Radek Krejcie7b95092019-05-15 11:03:07 +0200591 fsync(out->method.fd);
592 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200593 case LY_OUT_MEMORY:
594 case LY_OUT_CALLBACK:
Radek Krejcid3ca0632019-04-16 16:54:54 +0200595 /* nothing to do */
596 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200597 case LY_OUT_ERROR:
Radek Krejcia5bba312020-01-09 15:41:20 +0100598 LOGINT(NULL);
Radek Krejcid3ca0632019-04-16 16:54:54 +0200599 }
Radek Krejcie7b95092019-05-15 11:03:07 +0200600
601 free(out->buffered);
602 out->buf_size = out->buf_len = 0;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200603}
604
Radek Krejcibaeb8382020-05-27 16:44:53 +0200605API ssize_t
Radek Krejci241f6b52020-05-21 18:13:49 +0200606ly_write(struct ly_out *out, const char *buf, size_t len)
Radek Krejcid3ca0632019-04-16 16:54:54 +0200607{
608 int written = 0;
609
Radek Krejcibaeb8382020-05-27 16:44:53 +0200610 LYOUT_CHECK(out, -1 * out->status);
Radek Krejci56cc0872019-04-30 09:22:27 +0200611
Radek Krejcid3ca0632019-04-16 16:54:54 +0200612 if (out->hole_count) {
613 /* we are buffering data after a hole */
Radek Krejcie7b95092019-05-15 11:03:07 +0200614 if (out->buf_len + len > out->buf_size) {
615 out->buffered = ly_realloc(out->buffered, out->buf_len + len);
Radek Krejcid3ca0632019-04-16 16:54:54 +0200616 if (!out->buffered) {
617 out->buf_len = 0;
618 out->buf_size = 0;
Radek Krejcibaeb8382020-05-27 16:44:53 +0200619 LOGMEM(NULL);
620 return -LY_EMEM;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200621 }
Radek Krejcie7b95092019-05-15 11:03:07 +0200622 out->buf_size = out->buf_len + len;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200623 }
624
Radek Krejcie7b95092019-05-15 11:03:07 +0200625 memcpy(&out->buffered[out->buf_len], buf, len);
626 out->buf_len += len;
Radek Krejcibaeb8382020-05-27 16:44:53 +0200627 return len;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200628 }
629
Radek Krejci897ad2e2019-04-29 16:43:07 +0200630repeat:
Radek Krejcid3ca0632019-04-16 16:54:54 +0200631 switch (out->type) {
Radek Krejci241f6b52020-05-21 18:13:49 +0200632 case LY_OUT_MEMORY:
Radek Krejcie7b95092019-05-15 11:03:07 +0200633 if (out->method.mem.len + len + 1 > out->method.mem.size) {
Radek Krejcia5bba312020-01-09 15:41:20 +0100634 *out->method.mem.buf = ly_realloc(*out->method.mem.buf, out->method.mem.len + len + 1);
635 if (!*out->method.mem.buf) {
Radek Krejcid3ca0632019-04-16 16:54:54 +0200636 out->method.mem.len = 0;
637 out->method.mem.size = 0;
Radek Krejcibaeb8382020-05-27 16:44:53 +0200638 LOGMEM(NULL);
639 return -LY_EMEM;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200640 }
Radek Krejcie7b95092019-05-15 11:03:07 +0200641 out->method.mem.size = out->method.mem.len + len + 1;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200642 }
Radek Krejcia5bba312020-01-09 15:41:20 +0100643 memcpy(&(*out->method.mem.buf)[out->method.mem.len], buf, len);
Radek Krejcie7b95092019-05-15 11:03:07 +0200644 out->method.mem.len += len;
Radek Krejcia5bba312020-01-09 15:41:20 +0100645 (*out->method.mem.buf)[out->method.mem.len] = '\0';
Radek Krejci897ad2e2019-04-29 16:43:07 +0200646
Radek Krejcie7b95092019-05-15 11:03:07 +0200647 out->printed += len;
Radek Krejcibaeb8382020-05-27 16:44:53 +0200648 return len;
Radek Krejci241f6b52020-05-21 18:13:49 +0200649 case LY_OUT_FD:
Radek Krejcie7b95092019-05-15 11:03:07 +0200650 written = write(out->method.fd, buf, len);
Radek Krejcid3ca0632019-04-16 16:54:54 +0200651 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200652 case LY_OUT_FDSTREAM:
653 case LY_OUT_FILEPATH:
654 case LY_OUT_FILE:
Radek Krejcie7b95092019-05-15 11:03:07 +0200655 written = fwrite(buf, sizeof *buf, len, out->method.f);
Radek Krejcid3ca0632019-04-16 16:54:54 +0200656 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200657 case LY_OUT_CALLBACK:
Radek Krejcia5bba312020-01-09 15:41:20 +0100658 written = out->method.clb.func(out->method.clb.arg, buf, len);
Radek Krejcid3ca0632019-04-16 16:54:54 +0200659 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200660 case LY_OUT_ERROR:
Radek Krejcia5bba312020-01-09 15:41:20 +0100661 LOGINT(NULL);
Radek Krejcibaeb8382020-05-27 16:44:53 +0200662 return -LY_EINT;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200663 }
664
665 if (written < 0) {
Radek Krejci897ad2e2019-04-29 16:43:07 +0200666 if (errno == EAGAIN || errno == EWOULDBLOCK) {
667 goto repeat;
668 }
669 LOGERR(out->ctx, LY_ESYS, "%s: writing data failed (%s).", __func__, strerror(errno));
Radek Krejci56cc0872019-04-30 09:22:27 +0200670 out->status = LY_ESYS;
Radek Krejcibaeb8382020-05-27 16:44:53 +0200671 return -LY_ESYS;
Radek Krejcie7b95092019-05-15 11:03:07 +0200672 } else if ((size_t)written != len) {
673 LOGERR(out->ctx, LY_ESYS, "%s: writing data failed (unable to write %u from %u data).", __func__, len - (size_t)written, len);
Radek Krejci56cc0872019-04-30 09:22:27 +0200674 out->status = LY_ESYS;
Radek Krejcibaeb8382020-05-27 16:44:53 +0200675 return -LY_ESYS;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200676 } else {
Radek Krejci241f6b52020-05-21 18:13:49 +0200677 if (out->type == LY_OUT_FDSTREAM) {
Radek Krejcia5bba312020-01-09 15:41:20 +0100678 /* move the original file descriptor to the end of the output file */
679 lseek(out->method.fdstream.fd, 0, SEEK_END);
680 }
Radek Krejci897ad2e2019-04-29 16:43:07 +0200681 out->printed += written;
Radek Krejcibaeb8382020-05-27 16:44:53 +0200682 return written;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200683 }
684}
685
Radek Krejcibaeb8382020-05-27 16:44:53 +0200686ssize_t
Radek Krejci241f6b52020-05-21 18:13:49 +0200687ly_write_skip(struct ly_out *out, size_t count, size_t *position)
Radek Krejcid3ca0632019-04-16 16:54:54 +0200688{
Radek Krejcibaeb8382020-05-27 16:44:53 +0200689 LYOUT_CHECK(out, -1 * out->status);
Radek Krejci56cc0872019-04-30 09:22:27 +0200690
Radek Krejcid3ca0632019-04-16 16:54:54 +0200691 switch (out->type) {
Radek Krejci241f6b52020-05-21 18:13:49 +0200692 case LY_OUT_MEMORY:
Radek Krejcid3ca0632019-04-16 16:54:54 +0200693 if (out->method.mem.len + count > out->method.mem.size) {
Radek Krejcia5bba312020-01-09 15:41:20 +0100694 *out->method.mem.buf = ly_realloc(*out->method.mem.buf, out->method.mem.len + count);
695 if (!(*out->method.mem.buf)) {
Radek Krejcid3ca0632019-04-16 16:54:54 +0200696 out->method.mem.len = 0;
697 out->method.mem.size = 0;
Radek Krejcibaeb8382020-05-27 16:44:53 +0200698 out->status = LY_EMEM;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200699 LOGMEM_RET(NULL);
700 }
701 out->method.mem.size = out->method.mem.len + count;
702 }
703
704 /* save the current position */
705 *position = out->method.mem.len;
706
707 /* skip the memory */
708 out->method.mem.len += count;
Radek Krejci897ad2e2019-04-29 16:43:07 +0200709
710 /* update printed bytes counter despite we actually printed just a hole */
711 out->printed += count;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200712 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200713 case LY_OUT_FD:
714 case LY_OUT_FDSTREAM:
715 case LY_OUT_FILEPATH:
716 case LY_OUT_FILE:
717 case LY_OUT_CALLBACK:
Radek Krejcid3ca0632019-04-16 16:54:54 +0200718 /* buffer the hole */
719 if (out->buf_len + count > out->buf_size) {
720 out->buffered = ly_realloc(out->buffered, out->buf_len + count);
721 if (!out->buffered) {
722 out->buf_len = 0;
723 out->buf_size = 0;
Radek Krejcibaeb8382020-05-27 16:44:53 +0200724 out->status = LY_EMEM;
725 LOGMEM(NULL);
726 return -LY_EMEM;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200727 }
728 out->buf_size = out->buf_len + count;
729 }
730
731 /* save the current position */
732 *position = out->buf_len;
733
734 /* skip the memory */
735 out->buf_len += count;
736
737 /* increase hole counter */
738 ++out->hole_count;
Radek Krejcia5bba312020-01-09 15:41:20 +0100739
740 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200741 case LY_OUT_ERROR:
Radek Krejcia5bba312020-01-09 15:41:20 +0100742 LOGINT(NULL);
Radek Krejcibaeb8382020-05-27 16:44:53 +0200743 return -LY_EINT;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200744 }
745
Radek Krejcibaeb8382020-05-27 16:44:53 +0200746 return count;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200747}
748
Radek Krejcibaeb8382020-05-27 16:44:53 +0200749ssize_t
Radek Krejci241f6b52020-05-21 18:13:49 +0200750ly_write_skipped(struct ly_out *out, size_t position, const char *buf, size_t count)
Radek Krejcid3ca0632019-04-16 16:54:54 +0200751{
Radek Krejcibaeb8382020-05-27 16:44:53 +0200752 ssize_t ret = LY_SUCCESS;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200753
Radek Krejcibaeb8382020-05-27 16:44:53 +0200754 LYOUT_CHECK(out, -1 * out->status);
Radek Krejci56cc0872019-04-30 09:22:27 +0200755
Radek Krejcid3ca0632019-04-16 16:54:54 +0200756 switch (out->type) {
Radek Krejci241f6b52020-05-21 18:13:49 +0200757 case LY_OUT_MEMORY:
Radek Krejcid3ca0632019-04-16 16:54:54 +0200758 /* write */
Radek Krejcia5bba312020-01-09 15:41:20 +0100759 memcpy(&(*out->method.mem.buf)[position], buf, count);
Radek Krejcid3ca0632019-04-16 16:54:54 +0200760 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200761 case LY_OUT_FD:
762 case LY_OUT_FDSTREAM:
763 case LY_OUT_FILEPATH:
764 case LY_OUT_FILE:
765 case LY_OUT_CALLBACK:
Radek Krejcid3ca0632019-04-16 16:54:54 +0200766 if (out->buf_len < position + count) {
Radek Krejcibaeb8382020-05-27 16:44:53 +0200767 out->status = LY_EMEM;
768 LOGMEM(NULL);
769 return -LY_EMEM;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200770 }
771
772 /* write into the hole */
773 memcpy(&out->buffered[position], buf, count);
774
775 /* decrease hole counter */
776 --out->hole_count;
777
778 if (!out->hole_count) {
Radek Krejci897ad2e2019-04-29 16:43:07 +0200779 /* all holes filled, we can write the buffer,
780 * printed bytes counter is updated by ly_write() */
Radek Krejci241f6b52020-05-21 18:13:49 +0200781 ret = ly_write(out, out->buffered, out->buf_len);
Radek Krejcid3ca0632019-04-16 16:54:54 +0200782 out->buf_len = 0;
783 }
784 break;
Radek Krejci241f6b52020-05-21 18:13:49 +0200785 case LY_OUT_ERROR:
Radek Krejcia5bba312020-01-09 15:41:20 +0100786 LOGINT(NULL);
Radek Krejcibaeb8382020-05-27 16:44:53 +0200787 return -LY_EINT;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200788 }
789
Radek Krejci241f6b52020-05-21 18:13:49 +0200790 if (out->type == LY_OUT_FILEPATH) {
Radek Krejcia5bba312020-01-09 15:41:20 +0100791 /* move the original file descriptor to the end of the output file */
792 lseek(out->method.fdstream.fd, 0, SEEK_END);
793 }
Radek Krejcibaeb8382020-05-27 16:44:53 +0200794 return ret < 0 ? (-1 * ret) : LY_SUCCESS;
Radek Krejcid3ca0632019-04-16 16:54:54 +0200795}