]> git.cameronkatri.com Git - mandoc.git/blobdiff - demandoc.c
make the path to ar(1) configurable, needed by NixOS;
[mandoc.git] / demandoc.c
index da25f8f10c656b648f31cab9f0af62020e91b41f..57d0cc5f4120dee213091c638803161c87948ea2 100644 (file)
@@ -1,4 +1,4 @@
-/*     $Id: demandoc.c,v 1.1 2011/09/01 10:46:28 kristaps Exp $ */
+/*     $Id: demandoc.c,v 1.33 2019/03/03 11:01:15 schwarze Exp $ */
 /*
  * Copyright (c) 2011 Kristaps Dzonsons <kristaps@bsd.lv>
  *
  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  */
-#ifdef HAVE_CONFIG_H
 #include "config.h"
-#endif
+
+#include <sys/types.h>
 
 #include <assert.h>
-#include <getopt.h>
+#include <ctype.h>
 #include <stdio.h>
 #include <stdlib.h>
 #include <string.h>
+#include <unistd.h>
 
+#include "mandoc.h"
+#include "roff.h"
 #include "man.h"
 #include "mdoc.h"
-#include "mandoc.h"
+#include "mandoc_parse.h"
 
-static void     pline(int, int *, int *);
-static void     pman(const struct man_node *, int *, int *);
-static void     pmandoc(struct mparse *, int, const char *);
-static void     pmdoc(const struct mdoc_node *, int *, int *);
-static void     pstring(const char *, int, int *);
+static void     pline(int, int *, int *, int);
+static void     pman(const struct roff_node *, int *, int *, int);
+static void     pmandoc(struct mparse *, int, const char *, int);
+static void     pmdoc(const struct roff_node *, int *, int *, int);
+static void     pstring(const char *, int, int *, int);
 static void     usage(void);
 
 static const char       *progname;
@@ -41,140 +44,217 @@ int
 main(int argc, char *argv[])
 {
        struct mparse   *mp;
-       int              ch, i;
+       int              ch, fd, i, list;
        extern int       optind;
-       extern char     *optarg;
 
-       progname = strrchr(argv[0], '/');
-       if (progname == NULL)
+       if (argc < 1)
+               progname = "demandoc";
+       else if ((progname = strrchr(argv[0], '/')) == NULL)
                progname = argv[0];
        else
                ++progname;
 
        mp = NULL;
+       list = 0;
 
-       while (-1 != (ch = getopt(argc, argv, "")))
+       while (-1 != (ch = getopt(argc, argv, "ikm:pw")))
                switch (ch) {
+               case ('i'):
+                       /* FALLTHROUGH */
+               case ('k'):
+                       /* FALLTHROUGH */
+               case ('m'):
+                       /* FALLTHROUGH */
+               case ('p'):
+                       break;
+               case ('w'):
+                       list = 1;
+                       break;
                default:
                        usage();
-                       return((int)MANDOCLEVEL_BADARG);
+                       return (int)MANDOCLEVEL_BADARG;
                }
 
        argc -= optind;
        argv += optind;
 
-       mp = mparse_alloc(MPARSE_AUTO, MANDOCLEVEL_FATAL, NULL, NULL);
+       mchars_alloc();
+       mp = mparse_alloc(MPARSE_SO | MPARSE_UTF8 | MPARSE_LATIN1 |
+           MPARSE_VALIDATE, MANDOC_OS_OTHER, NULL);
        assert(mp);
 
-       if (0 == argc)
-               pmandoc(mp, STDIN_FILENO, "<stdin>");
+       if (argc < 1)
+               pmandoc(mp, STDIN_FILENO, "<stdin>", list);
 
        for (i = 0; i < argc; i++) {
                mparse_reset(mp);
-               pmandoc(mp, -1, argv[i]);
+               if ((fd = mparse_open(mp, argv[i])) == -1) {
+                       perror(argv[i]);
+                       continue;
+               }
+               pmandoc(mp, fd, argv[i], list);
        }
 
        mparse_free(mp);
-       return(MANDOCLEVEL_OK);
+       mchars_free();
+       return (int)MANDOCLEVEL_OK;
 }
 
 static void
 usage(void)
 {
 
-       fprintf(stderr, "usage: %s [files...]\n", progname);
+       fprintf(stderr, "usage: %s [-w] [files...]\n", progname);
 }
 
 static void
-pmandoc(struct mparse *mp, int fd, const char *fn)
+pmandoc(struct mparse *mp, int fd, const char *fn, int list)
 {
-       struct mdoc     *mdoc;
-       struct man      *man;
+       struct roff_meta        *meta;
        int              line, col;
 
-       if (mparse_readfd(mp, fd, fn) >= MANDOCLEVEL_FATAL) {
-               fprintf(stderr, "%s: Parse failure\n", fn);
-               return;
-       }
-
-       mparse_result(mp, &mdoc, &man);
+       mparse_readfd(mp, fd, fn);
+       close(fd);
+       meta = mparse_result(mp);
        line = 1;
        col = 0;
 
-       if (mdoc) 
-               pmdoc(mdoc_node(mdoc), &line, &col);
-       else if (man)
-               pman(man_node(man), &line, &col);
+       if (meta->macroset == MACROSET_MDOC)
+               pmdoc(meta->first->child, &line, &col, list);
        else
-               return;
+               pman(meta->first->child, &line, &col, list);
 
-       putchar('\n');
+       if ( ! list)
+               putchar('\n');
 }
 
 /*
  * Strip the escapes out of a string, emitting the results.
  */
 static void
-pstring(const char *p, int col, int *colp)
+pstring(const char *p, int col, int *colp, int list)
 {
        enum mandoc_esc  esc;
+       const char      *start, *end;
+       int              emit;
+
+       /*
+        * Print as many column spaces til we achieve parity with the
+        * input document.
+        */
+
+again:
+       if (list && '\0' != *p) {
+               while (isspace((unsigned char)*p))
+                       p++;
+
+               while ('\'' == *p || '(' == *p || '"' == *p)
+                       p++;
+
+               emit = isalpha((unsigned char)p[0]) &&
+                       isalpha((unsigned char)p[1]);
+
+               for (start = p; '\0' != *p; p++)
+                       if ('\\' == *p) {
+                               p++;
+                               esc = mandoc_escape(&p, NULL, NULL);
+                               if (ESCAPE_ERROR == esc)
+                                       return;
+                               emit = 0;
+                       } else if (isspace((unsigned char)*p))
+                               break;
+
+               end = p - 1;
+
+               while (end > start)
+                       if ('.' == *end || ',' == *end ||
+                                       '\'' == *end || '"' == *end ||
+                                       ')' == *end || '!' == *end ||
+                                       '?' == *end || ':' == *end ||
+                                       ';' == *end)
+                               end--;
+                       else
+                               break;
+
+               if (emit && end - start >= 1) {
+                       for ( ; start <= end; start++)
+                               if (ASCII_HYPH == *start)
+                                       putchar('-');
+                               else
+                                       putchar((unsigned char)*start);
+                       putchar('\n');
+               }
+
+               if (isspace((unsigned char)*p))
+                       goto again;
+
+               return;
+       }
 
        while (*colp < col) {
                putchar(' ');
                (*colp)++;
        }
 
-       while ('\0' != *p) {
+       /*
+        * Print the input word, skipping any special characters.
+        */
+       while ('\0' != *p)
                if ('\\' == *p) {
                        p++;
                        esc = mandoc_escape(&p, NULL, NULL);
                        if (ESCAPE_ERROR == esc)
-                               return;
+                               break;
                } else {
-                       putchar(*p++);
+                       putchar((unsigned char )*p++);
                        (*colp)++;
                }
-       }
 }
 
-/*
- * Emit lines until we're in sync with our input.
- */
 static void
-pline(int line, int *linep, int *col)
+pline(int line, int *linep, int *col, int list)
 {
 
+       if (list)
+               return;
+
+       /*
+        * Print out as many lines as needed to reach parity with the
+        * original input.
+        */
+
        while (*linep < line) {
                putchar('\n');
                (*linep)++;
        }
+
        *col = 0;
 }
 
 static void
-pmdoc(const struct mdoc_node *p, int *line, int *col)
+pmdoc(const struct roff_node *p, int *line, int *col, int list)
 {
 
        for ( ; p; p = p->next) {
-               if (MDOC_LINE & p->flags)
-                       pline(p->line, line, col);
-               if (MDOC_TEXT == p->type)
-                       pstring(p->string, p->pos, col);
-               if (p->child) 
-                       pmdoc(p->child, line, col);
+               if (NODE_LINE & p->flags)
+                       pline(p->line, line, col, list);
+               if (ROFFT_TEXT == p->type)
+                       pstring(p->string, p->pos, col, list);
+               if (p->child)
+                       pmdoc(p->child, line, col, list);
        }
 }
 
 static void
-pman(const struct man_node *p, int *line, int *col)
+pman(const struct roff_node *p, int *line, int *col, int list)
 {
 
        for ( ; p; p = p->next) {
-               if (MAN_LINE & p->flags)
-                       pline(p->line, line, col);
-               if (MAN_TEXT == p->type)
-                       pstring(p->string, p->pos, col);
-               if (p->child) 
-                       pman(p->child, line, col);
+               if (NODE_LINE & p->flags)
+                       pline(p->line, line, col, list);
+               if (ROFFT_TEXT == p->type)
+                       pstring(p->string, p->pos, col, list);
+               if (p->child)
+                       pman(p->child, line, col, list);
        }
 }