Graphviz 13.0.0~dev.20250402.0402
Loading...
Searching...
No Matches
gvusershape.c
Go to the documentation of this file.
1/*************************************************************************
2 * Copyright (c) 2011 AT&T Intellectual Property
3 * All rights reserved. This program and the accompanying materials
4 * are made available under the terms of the Eclipse Public License v1.0
5 * which accompanies this distribution, and is available at
6 * https://www.eclipse.org/legal/epl-v10.html
7 *
8 * Contributors: Details at https://graphviz.org
9 *************************************************************************/
10
11#include "config.h"
12#include <assert.h>
13#include <errno.h>
14#include <limits.h>
15#include <math.h>
16#include <stdbool.h>
17#include <stdio.h>
18#include <stdlib.h>
19#include <string.h>
20#include <util/gv_fopen.h>
21#include <util/optional.h>
22
23#ifdef _WIN32
24#include <windows.h>
25#define GLOB_NOSPACE 1 /* Ran out of memory. */
26#define GLOB_ABORTED 2 /* Read error. */
27#define GLOB_NOMATCH 3 /* No matches found. */
28#define GLOB_NOSORT 4
29#endif
30
31#include <common/types.h>
32#include <common/usershape.h>
33#include <common/utils.h>
34#include <gvc/gvcint.h>
35#include <gvc/gvcproc.h>
36#include <gvc/gvplugin.h>
38#include <util/agxbuf.h>
39#include <util/alloc.h>
40#include <util/gv_ctype.h>
41#include <util/strview.h>
42
43extern char *Gvimagepath;
44extern char *HTTPServerEnVar;
45extern shape_desc *find_user_shape(const char *);
46
48
49typedef struct {
50 char *template;
51 size_t size;
55
56#define HDRLEN 20
57
58#define PNG_MAGIC "\x89PNG\x0D\x0A\x1A\x0A"
59#define PS_MAGIC "%!PS-Adobe-"
60#define BMP_MAGIC "BM"
61#define GIF_MAGIC "GIF8"
62#define JPEG_MAGIC "\xFF\xD8\xFF"
63#define PDF_MAGIC "%PDF-"
64#define EPS_MAGIC "\xC5\xD0\xD3\xC6"
65#define XML_MAGIC "<?xml"
66#define SVG_MAGIC "<svg"
67#define RIFF_MAGIC "RIFF"
68#define WEBP_MAGIC "WEBP"
69#define ICO_MAGIC "\x00\x00\x01\x00"
70
72 {
74 sizeof(PNG_MAGIC) - 1,
75 FT_PNG,
76 "png",
77 },
78 {
80 sizeof(PS_MAGIC) - 1,
81 FT_PS,
82 "ps",
83 },
84 {
86 sizeof(BMP_MAGIC) - 1,
87 FT_BMP,
88 "bmp",
89 },
90 {
92 sizeof(GIF_MAGIC) - 1,
93 FT_GIF,
94 "gif",
95 },
96 {
98 sizeof(JPEG_MAGIC) - 1,
99 FT_JPEG,
100 "jpeg",
101 },
102 {
103 PDF_MAGIC,
104 sizeof(PDF_MAGIC) - 1,
105 FT_PDF,
106 "pdf",
107 },
108 {
109 EPS_MAGIC,
110 sizeof(EPS_MAGIC) - 1,
111 FT_EPS,
112 "eps",
113 },
114 {
115 XML_MAGIC,
116 sizeof(XML_MAGIC) - 1,
117 FT_XML,
118 "xml",
119 },
120 {
122 sizeof(RIFF_MAGIC) - 1,
123 FT_RIFF,
124 "riff",
125 },
126 {
127 ICO_MAGIC,
128 sizeof(ICO_MAGIC) - 1,
129 FT_ICO,
130 "ico",
131 },
132};
133
135 char header[HDRLEN] = {0};
136
137 if (us->f && fread(header, 1, HDRLEN, us->f) == HDRLEN) {
138 for (size_t i = 0; i < sizeof(knowntypes) / sizeof(knowntype_t); i++) {
139 if (!memcmp(header, knowntypes[i].template, knowntypes[i].size)) {
141 us->type = knowntypes[i].type;
142 if (us->type == FT_XML) {
143 // if we did not see the closing of the XML declaration, scan for it
144 if (memchr(header, '>', HDRLEN) == NULL) {
145 while (true) {
146 int c = fgetc(us->f);
147 if (c == EOF) {
148 return us->type;
149 } else if (c == '>') {
150 break;
151 }
152 }
153 }
154 /* check for SVG in case of XML */
155 char tag[sizeof(SVG_MAGIC) - 1] = {0};
156 if (fread(tag, 1, sizeof(tag), us->f) != sizeof(tag)) {
157 return us->type;
158 }
159 while (true) {
160 if (memcmp(tag, SVG_MAGIC, sizeof(SVG_MAGIC) - 1) == 0) {
161 us->stringtype = "svg";
162 return (us->type = FT_SVG);
163 }
164 int c = fgetc(us->f);
165 if (c == EOF) {
166 return us->type;
167 }
168 memmove(&tag[0], &tag[1], sizeof(tag) - 1);
169 tag[sizeof(tag) - 1] = (char)c;
170 }
171 } else if (us->type == FT_RIFF) {
172 /* check for WEBP in case of RIFF */
173 if (!memcmp(header + 8, WEBP_MAGIC, sizeof(WEBP_MAGIC) - 1)) {
174 us->stringtype = "webp";
175 return (us->type = FT_WEBP);
176 }
177 }
178 return us->type;
179 }
180 }
181 }
182
183 us->stringtype = "(lib)";
184 us->type = FT_NULL;
185
186 return FT_NULL;
187}
188
189static bool get_int_lsb_first(FILE *f, size_t sz, int *val) {
190 int ch;
191
192 unsigned value = 0;
193 for (size_t i = 0; i < sz; i++) {
194 ch = fgetc(f);
195 if (feof(f))
196 return false;
197 value |= (unsigned)ch << 8 * i;
198 }
199 if (value > INT_MAX) {
200 return false;
201 }
202 *val = (int)value;
203 return true;
204}
205
206static bool get_int_msb_first(FILE *f, size_t sz, int *val) {
207 int ch;
208
209 unsigned value = 0;
210 for (size_t i = 0; i < sz; i++) {
211 ch = fgetc(f);
212 if (feof(f))
213 return false;
214 value <<= 8;
215 value |= (unsigned)ch;
216 }
217 if (value > INT_MAX) {
218 return false;
219 }
220 *val = (int)value;
221 return true;
222}
223
224static double svg_units_convert(double n, char *u) {
225 if (strcmp(u, "in") == 0)
226 return round(n * POINTS_PER_INCH);
227 if (strcmp(u, "px") == 0)
228 return round(n * POINTS_PER_INCH / 96);
229 if (strcmp(u, "pc") == 0)
230 return round(n * POINTS_PER_INCH / 6);
231 if (strcmp(u, "pt") == 0 ||
232 strcmp(u, "\"") == 0) /* ugly!! - if there are no inits then the %2s get
233 the trailing '"' */
234 return round(n);
235 if (strcmp(u, "cm") == 0)
236 return round(n * POINTS_PER_CM);
237 if (strcmp(u, "mm") == 0)
238 return round(n * POINTS_PER_MM);
239 return 0;
240}
241
242typedef struct {
245} match_t;
246
247static int find_attribute(const char *s, match_t *result) {
248
249 // look for an attribute string matching ([a-z][a-zA-Z]*)="([^"]*)"
250 for (size_t i = 0; s[i] != '\0';) {
251 if (s[i] >= 'a' && s[i] <= 'z') {
252 result->key.data = &s[i];
253 result->key.size = 1;
254 ++i;
255 while ((s[i] >= 'a' && s[i] <= 'z') || (s[i] >= 'A' && s[i] <= 'Z')) {
256 ++i;
257 ++result->key.size;
258 }
259 if (s[i] == '=' && s[i + 1] == '"') {
260 i += 2;
261 result->value.data = &s[i];
262 result->value.size = 0;
263 while (s[i] != '"' && s[i] != '\0') {
264 ++i;
265 ++result->value.size;
266 }
267 if (s[i] == '"') {
268 // found a valid attribute
269 return 0;
270 }
271 }
272 } else {
273 ++i;
274 }
275 }
276
277 // no attribute found
278 return -1;
279}
280
281static void svg_size(usershape_t *us) {
282 double n;
283 char u[3];
284 agxbuf line = {0};
285 bool eof = false;
286
287 // authoritative constraints we learned from `height` and `width`
288 optional_double_t hard_height = {0};
289 optional_double_t hard_width = {0};
290
291 // fallback constraints we learned from `viewBox`
292 optional_double_t soft_height = {0};
293 optional_double_t soft_width = {0};
294
295 rewind(us->f);
296 while (!eof && (!hard_width.has_value || !hard_height.has_value)) {
297 // read next line
298 while (true) {
299 int c = fgetc(us->f);
300 if (c == EOF) {
301 eof = true;
302 break;
303 } else if (c == '\n') {
304 break;
305 }
306 agxbputc(&line, (char)c);
307 }
308
309 const char *re_string = agxbuse(&line);
311 while (find_attribute(re_string, &match) == 0) {
312 re_string = match.value.data + match.value.size + 1;
313
314 if (strview_str_eq(match.key, "width")) {
315 char *value = strview_str(match.value);
316 if (sscanf(value, "%lf%2s", &n, u) == 2) {
317 optional_double_set(&hard_width, svg_units_convert(n, u));
318 } else if (sscanf(value, "%lf", &n) == 1) {
319 optional_double_set(&hard_width, svg_units_convert(n, "pt"));
320 }
321 free(value);
322 if (hard_height.has_value)
323 break;
324 } else if (strview_str_eq(match.key, "height")) {
325 char *value = strview_str(match.value);
326 if (sscanf(value, "%lf%2s", &n, u) == 2) {
327 optional_double_set(&hard_height, svg_units_convert(n, u));
328 } else if (sscanf(value, "%lf", &n) == 1) {
329 optional_double_set(&hard_height, svg_units_convert(n, "pt"));
330 }
331 free(value);
332 if (hard_width.has_value)
333 break;
334 } else if (strview_str_eq(match.key, "viewBox")) {
335 char *value = strview_str(match.value);
336 double w, h;
337 if (sscanf(value, "%*f %*f %lf %lf", &w, &h) == 2) {
338 optional_double_set(&soft_width, w);
339 optional_double_set(&soft_height, h);
340 }
341 free(value);
342 }
343 }
344
345 // if we have reached the end of a line and have seen `viewBox` but not
346 // `height` and/or `width`, let `viewBox` determine the dimensions
347 if (soft_height.has_value && soft_width.has_value) {
348 if (!hard_height.has_value) {
349 optional_double_set(&hard_height, soft_height.value);
350 }
351 if (!hard_width.has_value) {
352 optional_double_set(&hard_width, soft_width.value);
353 }
354 break;
355 }
356 }
357 us->dpi = 0;
358 const double h = optional_double_value_or(hard_height, 0);
359 const double w = optional_double_value_or(hard_width, 0);
360 assert(w >= 0 && w <= INT_MAX);
361 us->w = (int)w;
362 assert(h >= 0 && h <= INT_MAX);
363 us->h = (int)h;
364 agxbfree(&line);
365}
366
367static void png_size(usershape_t *us) {
368 int w, h;
369
370 us->dpi = 0;
371 fseek(us->f, 16, SEEK_SET);
372 if (get_int_msb_first(us->f, 4, &w) && get_int_msb_first(us->f, 4, &h)) {
373 us->w = w;
374 us->h = h;
375 }
376}
377
378static void ico_size(usershape_t *us) {
379 int w, h;
380
381 us->dpi = 0;
382 fseek(us->f, 6, SEEK_SET);
383 if (get_int_msb_first(us->f, 1, &w) && get_int_msb_first(us->f, 1, &h)) {
384 us->w = w;
385 us->h = h;
386 }
387}
388
389static void webp_size(usershape_t *us) {
390 int w, h;
391
392 us->dpi = 0;
393 fseek(us->f, 15, SEEK_SET);
394 if (fgetc(us->f) == 'X') { // VP8X
395 fseek(us->f, 24, SEEK_SET);
396 if (get_int_lsb_first(us->f, 4, &w) && get_int_lsb_first(us->f, 4, &h)) {
397 us->w = w;
398 us->h = h;
399 }
400 } else { // VP8
401 fseek(us->f, 26, SEEK_SET);
402 if (get_int_lsb_first(us->f, 2, &w) && get_int_lsb_first(us->f, 2, &h)) {
403 us->w = w;
404 us->h = h;
405 }
406 }
407}
408
409static void gif_size(usershape_t *us) {
410 int w, h;
411
412 us->dpi = 0;
413 fseek(us->f, 6, SEEK_SET);
414 if (get_int_lsb_first(us->f, 2, &w) && get_int_lsb_first(us->f, 2, &h)) {
415 us->w = w;
416 us->h = h;
417 }
418}
419
420static void bmp_size(usershape_t *us) {
421 int size_x_msw, size_x_lsw, size_y_msw, size_y_lsw;
422
423 us->dpi = 0;
424 fseek(us->f, 16, SEEK_SET);
425 if (get_int_lsb_first(us->f, 2, &size_x_msw) &&
426 get_int_lsb_first(us->f, 2, &size_x_lsw) &&
427 get_int_lsb_first(us->f, 2, &size_y_msw) &&
428 get_int_lsb_first(us->f, 2, &size_y_lsw)) {
429 us->w = size_x_msw << 16 | size_x_lsw;
430 us->h = size_y_msw << 16 | size_y_lsw;
431 }
432}
433
434static void jpeg_size(usershape_t *us) {
435 int marker, length, size_x, size_y;
436
437 /* These are the markers that follow 0xff in the file.
438 * Other markers implicitly have a 2-byte length field that follows.
439 */
440 static const unsigned char standalone_markers[] = {
441 0x01, /* Temporary */
442 0xd0, 0xd1, 0xd2, 0xd3, /* Reset */
443 0xd4, 0xd5, 0xd6, 0xd7, 0xd8, /* Start of image */
444 0xd9, /* End of image */
445 };
446
447 us->dpi = 0;
448 rewind(us->f);
449 while (true) {
450 /* Now we must be at a 0xff or at a series of 0xff's.
451 * If that is not the case, or if we're at EOF, then there's
452 * a parsing error.
453 */
454 if (!get_int_msb_first(us->f, 1, &marker)) {
455 agwarningf("Parsing of \"%s\" failed\n", us->name);
456 return;
457 }
458
459 if (marker == 0xff)
460 continue;
461
462 /* Ok.. marker now read. If it is not a stand-alone marker,
463 * then continue. If it's a Start Of Frame (0xc?), then we're there.
464 * If it's another marker with a length field, then skip ahead
465 * over that length field.
466 */
467
468 /* A stand-alone... */
469 if (memchr(standalone_markers, marker, sizeof(standalone_markers)))
470 continue;
471
472 /* Incase of a 0xc0 marker: */
473 if (marker == 0xc0) {
474 /* Skip length and 2 lengths. */
475 if (fseek(us->f, 3, SEEK_CUR) == 0 &&
476 get_int_msb_first(us->f, 2, &size_x) &&
477 get_int_msb_first(us->f, 2, &size_y)) {
478
479 /* Store length. */
480 us->h = size_x;
481 us->w = size_y;
482 }
483 return;
484 }
485
486 /* Incase of a 0xc2 marker: */
487 if (marker == 0xc2) {
488 /* Skip length and one more byte */
489 if (fseek(us->f, 3, SEEK_CUR) != 0)
490 return;
491
492 /* Get length and store. */
493 if (get_int_msb_first(us->f, 2, &size_x) &&
494 get_int_msb_first(us->f, 2, &size_y)) {
495 us->h = size_x;
496 us->w = size_y;
497 }
498 return;
499 }
500
501 /* Any other marker is assumed to be followed by 2 bytes length. */
502 if (!get_int_msb_first(us->f, 2, &length))
503 return;
504
505 fseek(us->f, length - 2, SEEK_CUR);
506 }
507}
508
509static void ps_size(usershape_t *us) {
510 char line[BUFSIZ];
511 int lx, ly, ux, uy;
512 char *linep;
513
514 us->dpi = 72;
515 rewind(us->f);
516 bool saw_bb = false;
517 while (fgets(line, sizeof(line), us->f)) {
518 /* PostScript accepts \r as EOL, so using fgets () and looking for a
519 * bounding box comment at the beginning doesn't work in this case.
520 * As a heuristic, we first search for a bounding box comment in line.
521 * This obviously fails if not all of the numbers make it into the
522 * current buffer. This shouldn't be a problem, as the comment is
523 * typically near the beginning, and so should be read within the first
524 * BUFSIZ bytes (even on Windows where this is 512).
525 */
526 if (!(linep = strstr(line, "%%BoundingBox:")))
527 continue;
528 if (sscanf(linep, "%%%%BoundingBox: %d %d %d %d", &lx, &ly, &ux, &uy) ==
529 4) {
530 saw_bb = true;
531 break;
532 }
533 }
534 if (saw_bb) {
535 us->x = lx;
536 us->y = ly;
537 us->w = ux - lx;
538 us->h = uy - ly;
539 }
540}
541
542#define KEY "/MediaBox"
543
544typedef struct {
545 char *s;
546 char *buf;
547 FILE *fp;
548} stream_t;
549
550static char nxtc(stream_t *str) {
551 if (fgets(str->buf, BUFSIZ, str->fp)) {
552 str->s = str->buf;
553 return *(str->s);
554 }
555 return '\0';
556}
557
558#define strc(x) (*(x->s) ? *(x->s) : nxtc(x))
559#define stradv(x) (x->s++)
560
561static void skipWS(stream_t *str) {
562 char c;
563 while ((c = strc(str))) {
564 if (gv_isspace(c))
565 stradv(str);
566 else
567 return;
568 }
569}
570
571static int scanNum(char *tok, double *dp) {
572 char *endp;
573 double d = strtod(tok, &endp);
574
575 if (tok == endp)
576 return 1;
577 *dp = d;
578 return 0;
579}
580
581static void getNum(stream_t *str, char *buf) {
582 int len = 0;
583 char c;
584 skipWS(str);
585 while ((c = strc(str)) && (gv_isdigit(c) || (c == '.'))) {
586 buf[len++] = c;
587 stradv(str);
588 if (len == BUFSIZ - 1)
589 break;
590 }
591 buf[len] = '\0';
592
593 return;
594}
595
596static int boxof(stream_t *str, boxf *bp) {
597 char tok[BUFSIZ];
598
599 skipWS(str);
600 if (strc(str) != '[')
601 return 1;
602 stradv(str);
603 getNum(str, tok);
604 if (scanNum(tok, &bp->LL.x))
605 return 1;
606 getNum(str, tok);
607 if (scanNum(tok, &bp->LL.y))
608 return 1;
609 getNum(str, tok);
610 if (scanNum(tok, &bp->UR.x))
611 return 1;
612 getNum(str, tok);
613 if (scanNum(tok, &bp->UR.y))
614 return 1;
615 return 0;
616}
617
618static int bboxPDF(FILE *fp, boxf *bp) {
620 char *s;
621 char buf[BUFSIZ];
622 while (fgets(buf, BUFSIZ, fp)) {
623 if ((s = strstr(buf, KEY))) {
624 str.buf = buf;
625 str.s = s + (sizeof(KEY) - 1);
626 str.fp = fp;
627 return boxof(&str, bp);
628 }
629 }
630 return 1;
631}
632
633static void pdf_size(usershape_t *us) {
634 boxf bb;
635
636 us->dpi = 0;
637 rewind(us->f);
638 if (!bboxPDF(us->f, &bb)) {
639 us->x = bb.LL.x;
640 us->y = bb.LL.y;
641 us->w = bb.UR.x - bb.LL.x;
642 us->h = bb.UR.y - bb.LL.y;
643 }
644}
645
646static void usershape_close(void *p) {
647 usershape_t *us = p;
648
649 if (us->f)
650 fclose(us->f);
651 if (us->data && us->datafree)
652 us->datafree(us);
653 free(us);
654}
655
657 .key = offsetof(usershape_t, name),
658 .size = -1,
659 .freef = usershape_close,
660};
661
662usershape_t *gvusershape_find(const char *name) {
663 usershape_t *us;
664
665 assert(name);
666 assert(name[0]);
667
668 if (!ImageDict)
669 return NULL;
670
671 us = dtmatch(ImageDict, name);
672 return us;
673}
674
675#define MAX_USERSHAPE_FILES_OPEN 50
677 static int usershape_files_open_cnt;
678 const char *fn;
679
680 assert(us);
681 assert(us->name);
682 assert(us->name[0]);
683
684 if (us->f)
685 rewind(us->f);
686 else {
687 if (!(fn = safefile(us->name))) {
688 agwarningf("Filename \"%s\" is unsafe\n", us->name);
689 return false;
690 }
691 us->f = gv_fopen(fn, "rb");
692 if (us->f == NULL) {
693 agwarningf("%s while opening %s\n", strerror(errno), fn);
694 return false;
695 }
696 if (usershape_files_open_cnt >= MAX_USERSHAPE_FILES_OPEN)
697 us->nocache = true;
698 else
699 usershape_files_open_cnt++;
700 }
701 assert(us->f);
702 return true;
703}
704
706 if (us->nocache) {
707 if (us->f) {
708 fclose(us->f);
709 us->f = NULL;
710 }
711 }
712}
713
714static void freeUsershape(usershape_t *us) {
715 if (us->name)
716 agstrfree(0, us->name, false);
717 free(us);
718}
719
720static usershape_t *gvusershape_open(const char *name) {
721 usershape_t *us;
722
723 assert(name);
724
725 if (!ImageDict)
727
728 if (!(us = gvusershape_find(name))) {
729 us = gv_alloc(sizeof(usershape_t));
730
731 us->name = agstrdup(0, name);
732 if (!gvusershape_file_access(us)) {
733 freeUsershape(us);
734 return NULL;
735 }
736
737 assert(us->f);
738
739 switch (imagetype(us)) {
740 case FT_NULL:
741 if (!(us->data = find_user_shape(us->name))) {
743 "\"%s\" was not found as a file or as a shape library member\n",
744 us->name);
745 freeUsershape(us);
746 return NULL;
747 }
748 break;
749 case FT_GIF:
750 gif_size(us);
751 break;
752 case FT_PNG:
753 png_size(us);
754 break;
755 case FT_BMP:
756 bmp_size(us);
757 break;
758 case FT_JPEG:
759 jpeg_size(us);
760 break;
761 case FT_PS:
762 ps_size(us);
763 break;
764 case FT_WEBP:
765 webp_size(us);
766 break;
767 case FT_SVG:
768 svg_size(us);
769 break;
770 case FT_PDF:
771 pdf_size(us);
772 break;
773 case FT_ICO:
774 ico_size(us);
775 break;
776 case FT_EPS: /* no eps_size code available */
777 default:
778 break;
779 }
781 dtinsert(ImageDict, us);
782 return us;
783 }
785 return us;
786}
787
788/* gvusershape_size_dpi:
789 * Return image size in points.
790 */
792 point rv;
793
794 if (!us) {
795 rv.x = rv.y = -1;
796 } else {
797 if (us->dpi != 0) {
798 dpi.x = dpi.y = us->dpi;
799 }
800 rv.x = (int)(us->w * POINTS_PER_INCH / dpi.x);
801 rv.y = (int)(us->h * POINTS_PER_INCH / dpi.y);
802 }
803 return rv;
804}
805
806/* gvusershape_size:
807 * Loads user image from file name if not already loaded.
808 * Return image size in points.
809 */
811 point rv;
812 pointf dpi;
813 static char *oldpath;
814 usershape_t *us;
815
816 /* no shape file, no shape size */
817 if (!name || (*name == '\0')) {
818 rv.x = rv.y = -1;
819 return rv;
820 }
821
822 if (!HTTPServerEnVar && (oldpath != Gvimagepath)) {
823 oldpath = Gvimagepath;
824 if (ImageDict) {
826 ImageDict = NULL;
827 }
828 }
829
830 if ((dpi.y = GD_drawing(g)->dpi) >= 1.0)
831 dpi.x = dpi.y;
832 else
833 dpi.x = dpi.y = DEFAULT_DPI;
834
835 us = gvusershape_open(name);
836 rv = gvusershape_size_dpi(us, dpi);
837 return rv;
838}
size_t match(char *str, char *pat)
Definition actions.c:106
static void agxbfree(agxbuf *xb)
free any malloced resources
Definition agxbuf.h:78
static WUR char * agxbuse(agxbuf *xb)
Definition agxbuf.h:307
static int agxbputc(agxbuf *xb, char c)
add character to buffer
Definition agxbuf.h:277
Memory allocation wrappers that exit on failure.
static void * gv_alloc(size_t size)
Definition alloc.h:47
#define dtmatch(d, o)
Definition cdt.h:184
#define dtinsert(d, o)
Definition cdt.h:185
CDT_API Dtmethod_t * Dttree
Definition dttree.c:308
CDT_API int dtclose(Dt_t *)
Definition dtclose.c:8
CDT_API Dt_t * dtopen(Dtdisc_t *, Dtmethod_t *)
Definition dtopen.c:9
const char * safefile(const char *filename)
Definition utils.c:260
#define POINTS_PER_MM
Definition geom.h:60
#define POINTS_PER_CM
Definition geom.h:59
#define POINTS_PER_INCH
Definition geom.h:58
static double len(glCompPoint p)
Definition glutils.c:150
void free(void *)
node NULL
Definition grammar.y:163
void agwarningf(const char *fmt,...)
Definition agerror.c:173
#define GD_drawing(g)
Definition types.h:353
int agstrfree(Agraph_t *, const char *, bool is_html)
Definition refstr.c:413
char * agstrdup(Agraph_t *, const char *)
returns a pointer to a reference-counted copy of the argument string, creating one if necessary
Definition refstr.c:397
bool gvusershape_file_access(usershape_t *us)
replacements for ctype.h functions
static bool gv_isdigit(int c)
Definition gv_ctype.h:41
static bool gv_isspace(int c)
Definition gv_ctype.h:55
FILE * gv_fopen(const char *filename, const char *mode)
Definition gv_fopen.c:32
wrapper around fopen for internal library usage
static void ico_size(usershape_t *us)
static void freeUsershape(usershape_t *us)
#define XML_MAGIC
Definition gvusershape.c:65
static char nxtc(stream_t *str)
#define HDRLEN
Definition gvusershape.c:56
#define MAX_USERSHAPE_FILES_OPEN
static bool get_int_msb_first(FILE *f, size_t sz, int *val)
#define RIFF_MAGIC
Definition gvusershape.c:67
shape_desc * find_user_shape(const char *)
Definition shapes.c:3962
#define PNG_MAGIC
Definition gvusershape.c:58
static bool get_int_lsb_first(FILE *f, size_t sz, int *val)
#define JPEG_MAGIC
Definition gvusershape.c:62
static void skipWS(stream_t *str)
static Dtdisc_t ImageDictDisc
#define PS_MAGIC
Definition gvusershape.c:59
static int find_attribute(const char *s, match_t *result)
char * HTTPServerEnVar
Definition globals.h:53
static void webp_size(usershape_t *us)
#define BMP_MAGIC
Definition gvusershape.c:60
char * Gvimagepath
Definition globals.h:49
usershape_t * gvusershape_find(const char *name)
void gvusershape_file_release(usershape_t *us)
point gvusershape_size(graph_t *g, char *name)
#define KEY
point gvusershape_size_dpi(usershape_t *us, pointf dpi)
#define EPS_MAGIC
Definition gvusershape.c:64
#define ICO_MAGIC
Definition gvusershape.c:69
#define GIF_MAGIC
Definition gvusershape.c:61
static int boxof(stream_t *str, boxf *bp)
static void jpeg_size(usershape_t *us)
static usershape_t * gvusershape_open(const char *name)
static int bboxPDF(FILE *fp, boxf *bp)
static knowntype_t knowntypes[]
Definition gvusershape.c:71
static void png_size(usershape_t *us)
static void gif_size(usershape_t *us)
static imagetype_t imagetype(usershape_t *us)
static int scanNum(char *tok, double *dp)
#define PDF_MAGIC
Definition gvusershape.c:63
static Dict_t * ImageDict
Definition gvusershape.c:47
#define SVG_MAGIC
Definition gvusershape.c:66
static double svg_units_convert(double n, char *u)
static void pdf_size(usershape_t *us)
static void ps_size(usershape_t *us)
static void svg_size(usershape_t *us)
#define stradv(x)
static void bmp_size(usershape_t *us)
#define WEBP_MAGIC
Definition gvusershape.c:68
#define strc(x)
static void getNum(stream_t *str, char *buf)
static void usershape_close(void *p)
textitem scanner parser str
Definition htmlparse.y:224
C analog of C++’s std::optional
static void optional_double_set(optional_double_t *me, double value)
Definition optional.h:23
static double optional_double_value_or(optional_double_t me, double fallback)
Definition optional.h:34
graph or subgraph
Definition cgraph.h:424
Definition geom.h:41
pointf UR
Definition geom.h:41
pointf LL
Definition geom.h:41
Definition cdt.h:100
int key
Definition cdt.h:85
imagetype_t type
Definition gvusershape.c:52
size_t size
Definition gvusershape.c:51
char * stringtype
Definition gvusershape.c:53
strview_t value
strview_t key
a container that may or may not contain a double value
Definition optional.h:11
double value
the value if has_value is true
Definition optional.h:13
bool has_value
does this have a value?
Definition optional.h:12
Definition geom.h:27
int y
Definition geom.h:27
int x
Definition geom.h:27
double x
Definition geom.h:29
double y
Definition geom.h:29
char * buf
FILE * fp
char * s
a non-owning string reference
Definition strview.h:20
const char * data
start of the pointed to string
Definition strview.h:21
size_t size
extent of the string in bytes
Definition strview.h:22
const char * name
Definition usershape.h:54
char * stringtype
Definition usershape.h:60
FILE * f
Definition usershape.h:58
void(* datafree)(usershape_t *us)
Definition usershape.h:65
bool nocache
Definition usershape.h:57
void * data
Definition usershape.h:63
imagetype_t type
Definition usershape.h:59
double x
Definition usershape.h:61
double y
Definition usershape.h:61
double h
Definition usershape.h:61
double w
Definition usershape.h:61
Non-owning string references.
static bool strview_str_eq(strview_t a, const char *b)
compare a string reference to a string for equality
Definition strview.h:98
static char * strview_str(strview_t source)
make a heap-allocated string from this string view
Definition strview.h:41
static tok_t tok(const char *input, const char *separators)
begin tokenization of a new string
Definition tokenize.h:43
graphs, nodes and edges info: Agraphinfo_t, Agnodeinfo_t and Agedgeinfo_t
Definition grammar.c:93
imagetype_t
Definition usershape.h:24
@ FT_BMP
Definition usershape.h:25
@ FT_PS
Definition usershape.h:26
@ FT_ICO
Definition usershape.h:27
@ FT_NULL
Definition usershape.h:24
@ FT_EPS
Definition usershape.h:26
@ FT_SVG
Definition usershape.h:26
@ FT_WEBP
Definition usershape.h:27
@ FT_GIF
Definition usershape.h:25
@ FT_JPEG
Definition usershape.h:25
@ FT_RIFF
Definition usershape.h:27
@ FT_PNG
Definition usershape.h:25
@ FT_XML
Definition usershape.h:26
@ FT_PDF
Definition usershape.h:26