treeviz/treeviz.h
2025-03-13 22:51:06 -05:00

168 lines
5.2 KiB
C

#ifndef TREEVIZ_H
#define TREEVIZ_H
// TreeViz context. Contains settings for the visualizer functions. This
// is wrapped in a struct so you don't have to always pass the same options
// to every funcion call. However, if you want to do that, you can use the
// `TV_vizl` function.
typedef struct TVCtx_T {
// Output filename. Should end in `.pdf`
char *filename;
// Function that takes a reference to a node, and returns a string
// representing the node's data. The representation should not include any
// information about children, just the data stored in the node that the
// function was called on.
char *(*represent_data)(void *node);
// A function that takes a node, and iteratively returns each child of the
// node on each successive call, setting `done` to 1 when all children have
// been processed. Be sure to use `cur_idx` to track the function's progress:
// this function will be called in a recursive context, so using static memory
// introduces some funny bugs.
void *(*next_child)(void *root_node, int *done, int *cur_idx);
} TVCtx;
// Visualize the tree rooted at `root_node`, using the settings found in `ctx`.
void TV_vizv(void *root_node, TVCtx *ctx);
// Visualize the tree rooted at `root_node`, using the settings specifed. See
// the `TVCtx` struct.
void TV_vizl(void *root_node, char *filename,
char *(*represent_data)(void *root_node),
void *(*next_child)(void *root_node, int *done, int *cur_idx));
#endif
#ifdef TREEVIZ_IMPLEMENTATION
#ifndef TREEVIZ_IMPL_ALREADY
#define TREEVIZ_IMPL_ALREADY
#include <assert.h>
#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/wait.h>
#include <unistd.h>
#define ERRCHECK(...) \
if (errno != 0) { \
fprintf(stderr, "error: %s (errno %d)\n", strerror(errno), errno); \
fprintf(stderr, __VA_ARGS__); \
fputc('\n', stderr); \
assert(errno == 0); \
}
#define NOT_NULL(x) assert(x != NULL)
void _TV_write_dot(void *root_node, int fd, TVCtx *ctx, size_t *id) {
void *(*next_child)(void *root_node, int *done, int *cur_idx) =
ctx->next_child;
char *(*represent_data)(void *node) = ctx->represent_data;
char *repr = represent_data(root_node);
size_t cur_id = *id;
++(*id);
errno = 0;
if (root_node == NULL) {
dprintf(fd, "%zu [style=filled,fillcolor=gray,label=\"%s\"];", cur_id,
repr);
} else {
dprintf(fd, "%zu [label=\"%s\"];", cur_id, repr);
}
ERRCHECK("TreeViz: _TV_write_dot: could not write to pipe");
free(repr);
if (root_node == NULL) {
return;
}
int done = 0;
int iter = 0;
while (!done) {
void *child = next_child(root_node, &done, &iter);
errno = 0;
dprintf(fd, "%zu->%zu;", cur_id, *id);
ERRCHECK("TreeViz: _TV_write_dot: could not write to pipe");
_TV_write_dot(child, fd, ctx, id);
}
}
void TV_vizv(void *root_node, TVCtx *ctx) {
int pipe_stdin[2];
errno = 0;
pipe(pipe_stdin);
ERRCHECK("TreeViz: TV_vizv: could not create pipe to dot")
int child_stdin_write = pipe_stdin[1];
int child_stdin_read = pipe_stdin[0];
char o_argument[1024] = "-o";
strncat(o_argument, ctx->filename, 1021);
errno = 0;
pid_t pid = fork();
ERRCHECK("TreeViz: TV_vizv: could not fork process");
if (pid == 0) {
//// CHILD PROCESS
close(child_stdin_write);
ERRCHECK("TreeViz: TV_vizv: CHILD: could not close old pipe, write end");
dup2(child_stdin_read, STDIN_FILENO);
ERRCHECK("TreeViz: TV_vizv: CHILD: could not set stdin fileno");
close(child_stdin_read);
ERRCHECK("TreeViz: TV_vizv: CHILD: could not close old pipe, read end");
char *run = "dot";
execlp(run, run, "-Tpdf", o_argument);
ERRCHECK("TreeViz: TV_vizv: CHILD: could not exec `%s`.\n\nDo you have "
"graphviz installed and in your PATH?\n",
run);
} else {
//// PARENT PROCESS
close(child_stdin_read);
ERRCHECK("TreeViz: TV_vizv: could not close old pipe, read end");
dprintf(child_stdin_write, "digraph{node [shape=rect];");
ERRCHECK("TreeViz: TV_vizv: could not write to pipe");
size_t id = 0;
_TV_write_dot(root_node, child_stdin_write, ctx, &id);
errno = 0;
dprintf(child_stdin_write, "}");
ERRCHECK("TreeViz: TV_vizv: could not write to pipe");
close(child_stdin_write);
ERRCHECK("TreeViz: TV_vizv: could not close old pipe, write end");
int status;
wait(&status);
ERRCHECK("TreeViz: TV_vizv: could not wait on child process");
if (!WIFEXITED(status) || WEXITSTATUS(status) != 0) {
fprintf(stderr, "TreeViz: TV_vizv: dot process failed. "
"terminating immediately.\n");
assert(status == 0);
}
}
}
void TV_vizl(void *root_node, char *filename,
char *(*represent_data)(void *root_node),
void *(*next_child)(void *root_node, int *done, int *cur_idx)) {
TVCtx ctx = {filename, represent_data, next_child};
TV_vizv(root_node, &ctx);
}
#endif
#endif