1998-02-27 07:54:42 +03:00
|
|
|
/* Man page to help file converter
|
|
|
|
Copyright (C) 1994, 1995 Janne Kukonlehto
|
2002-09-01 14:45:35 +04:00
|
|
|
2002 Andrew V. Samoilov
|
|
|
|
2002 Pavel Roskin
|
|
|
|
|
1998-02-27 07:54:42 +03:00
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
2000-08-23 02:50:00 +04:00
|
|
|
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
|
1998-02-27 07:54:42 +03:00
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <string.h>
|
2002-07-31 02:20:26 +04:00
|
|
|
|
2002-10-15 19:38:15 +04:00
|
|
|
#include <glib.h>
|
1998-02-27 07:54:42 +03:00
|
|
|
#include "help.h"
|
|
|
|
|
|
|
|
#define BUFFER_SIZE 256
|
|
|
|
|
|
|
|
static int col = 0; /* Current output column */
|
|
|
|
static int out_row = 1; /* Current output row */
|
|
|
|
static int in_row = 0; /* Current input row */
|
|
|
|
static int no_split_flag = 0; /* Flag: Don't split section on next ".SH" */
|
|
|
|
static int skip_flag = 0; /* Flag: Skip this section.
|
|
|
|
0 = don't skip,
|
|
|
|
1 = skipping title,
|
|
|
|
2 = title skipped, skipping text */
|
|
|
|
static int link_flag = 0; /* Flag: Next line is a link */
|
|
|
|
static int verbatim_flag = 0; /* Flag: Copy input to output verbatim */
|
2002-02-28 17:38:38 +03:00
|
|
|
static int node = 0; /* Flag: This line is an original ".SH" */
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2002-08-20 20:13:13 +04:00
|
|
|
static const char *c_out; /* Output filename */
|
|
|
|
static FILE *f_out; /* Output file */
|
|
|
|
|
2002-09-01 14:45:35 +04:00
|
|
|
static const char *c_in; /* Current input filename */
|
|
|
|
|
2002-08-27 08:27:26 +04:00
|
|
|
static char *topics = NULL;
|
2002-07-31 02:20:26 +04:00
|
|
|
|
2002-09-02 06:06:44 +04:00
|
|
|
struct node {
|
2002-10-15 18:41:27 +04:00
|
|
|
char *node; /* Section name */
|
|
|
|
char *lname; /* Translated .SH, NULL if not translated */
|
2002-07-31 02:20:26 +04:00
|
|
|
struct node *next;
|
2002-10-15 18:41:27 +04:00
|
|
|
int heading_level;
|
2002-09-02 06:06:44 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct node nodes;
|
2002-10-15 18:41:27 +04:00
|
|
|
static struct node *cnode; /* Current node */
|
2002-07-31 02:20:26 +04:00
|
|
|
|
2002-08-16 20:48:27 +04:00
|
|
|
#define MAX_STREAM_BLOCK 8192
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Read in blocks of reasonable size and make sure we read everything.
|
|
|
|
* Failure to read everything is an error.
|
|
|
|
*/
|
|
|
|
static size_t
|
|
|
|
persistent_fread (void *data, size_t len, FILE * stream)
|
|
|
|
{
|
|
|
|
size_t count;
|
|
|
|
size_t bytes_done = 0;
|
|
|
|
char *ptr = (char *) data;
|
|
|
|
|
|
|
|
while (bytes_done < len) {
|
|
|
|
count = len - bytes_done;
|
|
|
|
if (count > MAX_STREAM_BLOCK)
|
|
|
|
count = MAX_STREAM_BLOCK;
|
|
|
|
|
|
|
|
count = fread (ptr, 1, count, stream);
|
|
|
|
|
|
|
|
if (count <= 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
bytes_done += count;
|
|
|
|
ptr += count;
|
|
|
|
}
|
|
|
|
|
|
|
|
return bytes_done;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Write in blocks of reasonable size and make sure we write everything.
|
|
|
|
* Failure to write everything is an error.
|
|
|
|
*/
|
|
|
|
static size_t
|
|
|
|
persistent_fwrite (const void *data, size_t len, FILE * stream)
|
|
|
|
{
|
|
|
|
size_t count;
|
|
|
|
size_t bytes_done = 0;
|
|
|
|
const char *ptr = (const char *) data;
|
|
|
|
|
|
|
|
while (bytes_done < len) {
|
|
|
|
count = len - bytes_done;
|
|
|
|
if (count > MAX_STREAM_BLOCK)
|
|
|
|
count = MAX_STREAM_BLOCK;
|
|
|
|
|
|
|
|
count = fwrite (ptr, 1, count, stream);
|
|
|
|
|
|
|
|
if (count <= 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
bytes_done += count;
|
|
|
|
ptr += count;
|
|
|
|
}
|
|
|
|
|
|
|
|
return bytes_done;
|
|
|
|
}
|
|
|
|
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Report error in input */
|
2002-07-31 02:57:01 +04:00
|
|
|
static void
|
2002-08-27 14:39:09 +04:00
|
|
|
print_error (const char *message)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2002-09-02 06:06:44 +04:00
|
|
|
fprintf (stderr, "man2hlp: %s in file \"%s\" on line %d\n", message,
|
2002-09-01 14:45:35 +04:00
|
|
|
c_in, in_row);
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
|
2002-08-27 08:27:26 +04:00
|
|
|
/* Do fopen(), exit if it fails */
|
|
|
|
static FILE *
|
|
|
|
fopen_check (const char *filename, const char *flags)
|
|
|
|
{
|
2002-08-27 14:39:09 +04:00
|
|
|
char tmp[BUFFER_SIZE];
|
2002-08-27 08:27:26 +04:00
|
|
|
FILE *f;
|
|
|
|
|
|
|
|
f = fopen (filename, flags);
|
|
|
|
if (f == NULL) {
|
2002-10-15 19:38:15 +04:00
|
|
|
g_snprintf (tmp, sizeof (tmp), "man2hlp: Cannot open file \"%s\"",
|
|
|
|
filename);
|
2002-08-27 08:27:26 +04:00
|
|
|
perror (tmp);
|
|
|
|
exit (3);
|
|
|
|
}
|
|
|
|
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Do fclose(), exit if it fails */
|
|
|
|
static void
|
|
|
|
fclose_check (FILE * f)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = fclose (f);
|
|
|
|
if (ret != 0) {
|
|
|
|
perror ("man2hlp: Cannot close file");
|
|
|
|
exit (3);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Change output line */
|
2002-07-31 02:57:01 +04:00
|
|
|
static void
|
|
|
|
newline (void)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2002-07-31 02:57:01 +04:00
|
|
|
out_row++;
|
1998-02-27 07:54:42 +03:00
|
|
|
col = 0;
|
2002-08-20 20:13:13 +04:00
|
|
|
fprintf (f_out, "\n");
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Calculate the length of string */
|
2002-07-31 02:57:01 +04:00
|
|
|
static int
|
2002-10-15 18:41:27 +04:00
|
|
|
string_len (const char *buffer)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
|
|
|
static int anchor_flag = 0; /* Flag: Inside hypertext anchor name */
|
|
|
|
static int link_flag = 0; /* Flag: Inside hypertext link target name */
|
|
|
|
int backslash_flag = 0; /* Flag: Backslash quoting */
|
|
|
|
int c; /* Current character */
|
|
|
|
int len = 0; /* Result: the length of the string */
|
|
|
|
|
2002-07-31 02:57:01 +04:00
|
|
|
while (*(buffer)) {
|
2002-02-28 17:12:34 +03:00
|
|
|
c = *buffer++;
|
2002-07-31 02:57:01 +04:00
|
|
|
if (c == CHAR_LINK_POINTER)
|
1998-02-27 07:54:42 +03:00
|
|
|
link_flag = 1; /* Link target name starts */
|
|
|
|
else if (c == CHAR_LINK_END)
|
|
|
|
link_flag = 0; /* Link target name ends */
|
2002-07-31 02:57:01 +04:00
|
|
|
else if (c == CHAR_NODE_END) {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Node anchor name starts */
|
|
|
|
anchor_flag = 1;
|
|
|
|
/* Ugly hack to prevent loss of one space */
|
2002-07-31 02:57:01 +04:00
|
|
|
len++;
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
/* Don't add control characters to the length */
|
2002-02-22 08:54:00 +03:00
|
|
|
if (c >= 0 && c < 32)
|
1998-02-27 07:54:42 +03:00
|
|
|
continue;
|
|
|
|
/* Attempt to handle backslash quoting */
|
2002-07-31 02:57:01 +04:00
|
|
|
if (c == '\\' && !backslash_flag) {
|
1998-02-27 07:54:42 +03:00
|
|
|
backslash_flag = 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
backslash_flag = 0;
|
|
|
|
/* Increase length if not inside anchor name or link target name */
|
|
|
|
if (!anchor_flag && !link_flag)
|
2002-07-31 02:57:01 +04:00
|
|
|
len++;
|
|
|
|
if (anchor_flag && c == ']') {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Node anchor name ends */
|
|
|
|
anchor_flag = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Output the string */
|
2002-07-31 02:57:01 +04:00
|
|
|
static void
|
|
|
|
print_string (char *buffer)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2002-07-31 02:57:01 +04:00
|
|
|
int len; /* The length of current word */
|
|
|
|
int c; /* Current character */
|
1998-02-27 07:54:42 +03:00
|
|
|
int backslash_flag = 0;
|
|
|
|
|
|
|
|
/* Skipping lines? */
|
|
|
|
if (skip_flag)
|
|
|
|
return;
|
|
|
|
/* Copying verbatim? */
|
2002-07-31 02:57:01 +04:00
|
|
|
if (verbatim_flag) {
|
2001-08-16 05:36:41 +04:00
|
|
|
/* Attempt to handle backslash quoting */
|
2002-07-31 02:57:01 +04:00
|
|
|
while (*(buffer)) {
|
2002-02-28 17:12:34 +03:00
|
|
|
c = *buffer++;
|
2002-07-31 02:57:01 +04:00
|
|
|
if (c == '\\' && !backslash_flag) {
|
2001-08-16 05:36:41 +04:00
|
|
|
backslash_flag = 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
backslash_flag = 0;
|
2002-08-20 20:13:13 +04:00
|
|
|
fprintf (f_out, "%c", c);
|
2001-08-16 05:36:41 +04:00
|
|
|
}
|
|
|
|
} else {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Split into words */
|
|
|
|
buffer = strtok (buffer, " \t\n");
|
|
|
|
/* Repeat for each word */
|
2002-07-31 02:57:01 +04:00
|
|
|
while (buffer) {
|
|
|
|
/* Skip empty strings */
|
|
|
|
if (*(buffer)) {
|
1998-02-27 07:54:42 +03:00
|
|
|
len = string_len (buffer);
|
|
|
|
/* Change the line if about to break the right margin */
|
2002-09-22 21:03:28 +04:00
|
|
|
if (col + len >= HELP_TEXT_WIDTH)
|
1998-02-27 07:54:42 +03:00
|
|
|
newline ();
|
|
|
|
/* Words are separated by spaces */
|
2002-07-31 02:57:01 +04:00
|
|
|
if (col > 0) {
|
2002-08-20 20:13:13 +04:00
|
|
|
fprintf (f_out, " ");
|
2002-07-31 02:57:01 +04:00
|
|
|
col++;
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
/* Attempt to handle backslash quoting */
|
2002-07-31 02:57:01 +04:00
|
|
|
while (*(buffer)) {
|
2002-02-28 17:38:38 +03:00
|
|
|
c = *buffer++;
|
2002-07-31 02:57:01 +04:00
|
|
|
if (c == '\\' && !backslash_flag) {
|
1998-02-27 07:54:42 +03:00
|
|
|
backslash_flag = 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
backslash_flag = 0;
|
2002-08-20 20:13:13 +04:00
|
|
|
fprintf (f_out, "%c", c);
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
/* Increase column */
|
|
|
|
col += len;
|
|
|
|
}
|
|
|
|
/* Get the next word */
|
|
|
|
buffer = strtok (NULL, " \t\n");
|
2002-07-31 02:57:01 +04:00
|
|
|
} /* while */
|
2001-08-16 05:36:41 +04:00
|
|
|
}
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Like print_string but with printf-like syntax */
|
2002-07-31 02:57:01 +04:00
|
|
|
static void
|
2002-09-01 14:45:35 +04:00
|
|
|
printf_string (const char *format, ...)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
|
|
|
va_list args;
|
2002-07-31 02:57:01 +04:00
|
|
|
char buffer[BUFFER_SIZE];
|
1998-02-27 07:54:42 +03:00
|
|
|
|
|
|
|
va_start (args, format);
|
2002-10-15 19:38:15 +04:00
|
|
|
g_vsnprintf (buffer, sizeof (buffer), format, args);
|
1998-02-27 07:54:42 +03:00
|
|
|
va_end (args);
|
|
|
|
print_string (buffer);
|
|
|
|
}
|
|
|
|
|
2002-09-02 06:06:44 +04:00
|
|
|
/* Handle NODE and .SH commands. is_sh is 1 for .SH, 0 for NODE */
|
2002-07-31 02:57:01 +04:00
|
|
|
static void
|
2002-09-02 06:06:44 +04:00
|
|
|
handle_node (char *buffer, int is_sh)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2001-08-16 05:36:41 +04:00
|
|
|
int len, heading_level;
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2002-09-02 06:06:44 +04:00
|
|
|
/* If we already skipped a section, don't skip another */
|
|
|
|
if (skip_flag == 2) {
|
|
|
|
skip_flag = 0;
|
|
|
|
}
|
|
|
|
/* Get the command parameters */
|
|
|
|
buffer = strtok (NULL, "");
|
|
|
|
if (buffer == NULL) {
|
|
|
|
print_error ("Syntax error: .SH: no title");
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
/* Remove quotes */
|
|
|
|
if (buffer[0] == '"') {
|
|
|
|
buffer++;
|
|
|
|
len = strlen (buffer);
|
|
|
|
if (buffer[len - 1] == '"') {
|
|
|
|
len--;
|
|
|
|
buffer[len] = 0;
|
|
|
|
}
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
2002-09-02 06:06:44 +04:00
|
|
|
/* Calculate heading level */
|
|
|
|
heading_level = 0;
|
|
|
|
while (buffer[heading_level] == ' ')
|
|
|
|
heading_level++;
|
|
|
|
/* Heading level must be even */
|
|
|
|
if (heading_level & 1)
|
|
|
|
print_error ("Syntax error: .SH: odd heading level");
|
|
|
|
if (no_split_flag) {
|
|
|
|
/* Don't start a new section */
|
|
|
|
newline ();
|
|
|
|
print_string (buffer);
|
|
|
|
newline ();
|
|
|
|
newline ();
|
|
|
|
no_split_flag = 0;
|
|
|
|
} else if (skip_flag) {
|
|
|
|
/* Skipping title and marking text for skipping */
|
|
|
|
skip_flag = 2;
|
1998-02-27 07:54:42 +03:00
|
|
|
} else {
|
2002-10-15 18:41:27 +04:00
|
|
|
buffer += heading_level;
|
2002-09-02 06:06:44 +04:00
|
|
|
if (!is_sh || !node) {
|
|
|
|
/* Start a new section, but omit empty section names */
|
2002-10-15 18:41:27 +04:00
|
|
|
if (*buffer) {
|
|
|
|
fprintf (f_out, "%c[%s]", CHAR_NODE_END, buffer);
|
2002-09-02 06:06:44 +04:00
|
|
|
col++;
|
|
|
|
newline ();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add section to the linked list */
|
|
|
|
if (!cnode) {
|
|
|
|
cnode = &nodes;
|
|
|
|
} else {
|
|
|
|
cnode->next = malloc (sizeof (nodes));
|
|
|
|
cnode = cnode->next;
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
2002-09-02 06:06:44 +04:00
|
|
|
cnode->node = strdup (buffer);
|
|
|
|
cnode->lname = NULL;
|
2002-10-15 18:41:27 +04:00
|
|
|
cnode->next = NULL;
|
|
|
|
cnode->heading_level = heading_level;
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
2002-09-02 06:06:44 +04:00
|
|
|
if (is_sh) {
|
|
|
|
/* print_string() strtok()es buffer, so */
|
2002-10-15 18:41:27 +04:00
|
|
|
cnode->lname = strdup (buffer);
|
|
|
|
print_string (buffer);
|
2001-08-16 05:36:41 +04:00
|
|
|
newline ();
|
|
|
|
newline ();
|
2002-09-02 06:06:44 +04:00
|
|
|
}
|
|
|
|
} /* Start new section */
|
|
|
|
} /* Has parameters */
|
|
|
|
node = !is_sh;
|
|
|
|
}
|
|
|
|
|
2002-09-22 23:19:45 +04:00
|
|
|
/* Convert character from the macro name to the font marker */
|
|
|
|
static inline char
|
|
|
|
char_to_font (char c)
|
|
|
|
{
|
|
|
|
switch (c) {
|
|
|
|
case 'R':
|
|
|
|
return CHAR_FONT_NORMAL;
|
|
|
|
case 'B':
|
|
|
|
return CHAR_FONT_BOLD;
|
|
|
|
case 'I':
|
|
|
|
return CHAR_FONT_ITALIC;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Handle alternate font commands (.BR, .IR, .RB, .RI, .BI, .IB)
|
|
|
|
* Return 0 if the command wasn't recognized, 1 otherwise
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
handle_alt_font (char *buffer)
|
|
|
|
{
|
|
|
|
char *p;
|
|
|
|
char *w;
|
|
|
|
char font[2];
|
|
|
|
int in_quotes = 0;
|
|
|
|
int alt_state = 0;
|
|
|
|
|
|
|
|
if (strlen (buffer) != 3)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (buffer[0] != '.')
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
font[0] = char_to_font (buffer[1]);
|
|
|
|
font[1] = char_to_font (buffer[2]);
|
|
|
|
|
|
|
|
/* Exclude names with unknown characters, .BB, .II and .RR */
|
|
|
|
if (font[0] == 0 || font[1] == 0 || font[0] == font[1])
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
p = strtok (NULL, "");
|
|
|
|
if (p == NULL) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
w = buffer;
|
|
|
|
*w++ = font[0];
|
|
|
|
|
|
|
|
while (*p) {
|
|
|
|
|
|
|
|
if (*p == '"') {
|
|
|
|
in_quotes = !in_quotes;
|
|
|
|
p++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*p == ' ' && !in_quotes) {
|
|
|
|
p++;
|
|
|
|
/* Don't change font if we are at the end */
|
|
|
|
if (*p != 0) {
|
|
|
|
alt_state = !alt_state;
|
|
|
|
*w++ = font[alt_state];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Skip more spaces */
|
|
|
|
while (*p == ' ')
|
|
|
|
p++;
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
*w++ = *p++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Turn off attributes if necessary */
|
|
|
|
if (font[alt_state] != CHAR_FONT_NORMAL)
|
|
|
|
*w++ = CHAR_FONT_NORMAL;
|
|
|
|
|
|
|
|
*w = 0;
|
|
|
|
print_string (buffer);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2002-09-02 06:06:44 +04:00
|
|
|
/* Handle all the roff dot commands. See man groff_man for details */
|
|
|
|
static void
|
|
|
|
handle_command (char *buffer)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
|
|
|
|
/* Get the command name */
|
|
|
|
strtok (buffer, " \t");
|
|
|
|
|
|
|
|
if (strcmp (buffer, ".SH") == 0) {
|
|
|
|
handle_node (buffer, 1);
|
|
|
|
} else if (strcmp (buffer, ".\\\"NODE") == 0) {
|
|
|
|
handle_node (buffer, 0);
|
|
|
|
} else if (strcmp (buffer, ".\\\"DONT_SPLIT\"") == 0) {
|
1998-02-27 07:54:42 +03:00
|
|
|
no_split_flag = 1;
|
2002-07-31 02:57:01 +04:00
|
|
|
} else if (strcmp (buffer, ".\\\"SKIP_SECTION\"") == 0) {
|
1998-02-27 07:54:42 +03:00
|
|
|
skip_flag = 1;
|
2002-07-31 02:57:01 +04:00
|
|
|
} else if (strcmp (buffer, ".\\\"LINK2\"") == 0) {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Next two input lines form a link */
|
|
|
|
link_flag = 2;
|
2002-09-01 14:45:35 +04:00
|
|
|
} else if ((strcmp (buffer, ".PP") == 0) || (strcmp (buffer, ".P") == 0)
|
|
|
|
|| (strcmp (buffer, ".LP") == 0)) {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* End of paragraph */
|
2002-07-31 02:57:01 +04:00
|
|
|
if (col > 0)
|
|
|
|
newline ();
|
1998-02-27 07:54:42 +03:00
|
|
|
newline ();
|
2002-07-31 02:57:01 +04:00
|
|
|
} else if (strcmp (buffer, ".nf") == 0) {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Following input lines are to be handled verbatim */
|
|
|
|
verbatim_flag = 1;
|
2002-07-31 02:57:01 +04:00
|
|
|
if (col > 0)
|
|
|
|
newline ();
|
2002-09-01 20:54:11 +04:00
|
|
|
} else if (strcmp (buffer, ".I") == 0 || strcmp (buffer, ".B") == 0
|
|
|
|
|| strcmp (buffer, ".SB") == 0) {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Bold text or italics text */
|
2002-03-21 11:13:58 +03:00
|
|
|
char *p;
|
2002-09-01 20:54:11 +04:00
|
|
|
char *w;
|
2002-03-21 11:13:58 +03:00
|
|
|
int backslash_flag = 0;
|
1998-02-27 07:54:42 +03:00
|
|
|
|
2002-09-01 20:54:11 +04:00
|
|
|
/* .SB [text]
|
|
|
|
* Causes the text on the same line or the text on the
|
|
|
|
* next line to appear in boldface font, one point
|
|
|
|
* size smaller than the default font.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* FIXME: text is optional, so there is no error */
|
|
|
|
p = strtok (NULL, "");
|
|
|
|
if (p == NULL) {
|
|
|
|
print_error ("Syntax error: .I | .B | .SB : no text");
|
1998-02-27 07:54:42 +03:00
|
|
|
return;
|
|
|
|
}
|
2002-03-21 11:13:58 +03:00
|
|
|
|
2002-09-22 20:40:33 +04:00
|
|
|
*buffer = (buffer[1] == 'I') ? CHAR_FONT_ITALIC : CHAR_FONT_BOLD;
|
2002-03-21 11:13:58 +03:00
|
|
|
|
|
|
|
/* Attempt to handle backslash quoting */
|
2002-09-01 20:54:11 +04:00
|
|
|
for (w = &buffer[1]; *p; p++) {
|
2002-07-31 02:57:01 +04:00
|
|
|
if (*p == '\\' && !backslash_flag) {
|
2002-03-21 11:13:58 +03:00
|
|
|
backslash_flag = 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
backslash_flag = 0;
|
|
|
|
*w++ = *p;
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
2002-03-21 11:13:58 +03:00
|
|
|
|
2002-09-22 20:40:33 +04:00
|
|
|
*w++ = CHAR_FONT_NORMAL;
|
2002-03-21 11:13:58 +03:00
|
|
|
*w = 0;
|
|
|
|
print_string (buffer);
|
2002-09-01 20:54:11 +04:00
|
|
|
} else if ((strcmp (buffer, ".TP") == 0)
|
|
|
|
|| (strcmp (buffer, ".IP") == 0)) {
|
|
|
|
/* TODO: Implement these indented paragraphs */
|
2002-07-31 02:57:01 +04:00
|
|
|
if (col > 0)
|
|
|
|
newline ();
|
1998-02-27 07:54:42 +03:00
|
|
|
newline ();
|
2002-07-31 02:57:01 +04:00
|
|
|
} else if (strcmp (buffer, ".\\\"TOPICS") == 0) {
|
|
|
|
if (out_row > 1) {
|
2002-09-01 20:54:11 +04:00
|
|
|
print_error
|
|
|
|
("Syntax error: .\\\"TOPICS must be first command");
|
2002-03-20 17:03:55 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
buffer = strtok (NULL, "");
|
2002-07-31 02:57:01 +04:00
|
|
|
if (buffer == NULL) {
|
2002-03-20 17:03:55 +03:00
|
|
|
print_error ("Syntax error: .\\\"TOPICS: no text");
|
|
|
|
return;
|
|
|
|
}
|
2002-08-25 22:13:37 +04:00
|
|
|
/* Remove quotes */
|
|
|
|
if (buffer[0] == '"') {
|
|
|
|
buffer++;
|
|
|
|
len = strlen (buffer);
|
|
|
|
if (buffer[len - 1] == '"') {
|
|
|
|
len--;
|
|
|
|
buffer[len] = 0;
|
|
|
|
}
|
|
|
|
}
|
2002-08-27 08:27:26 +04:00
|
|
|
topics = strdup (buffer);
|
2002-09-01 14:45:35 +04:00
|
|
|
} else if (strcmp (buffer, ".br") == 0) {
|
|
|
|
if (col)
|
|
|
|
newline ();
|
|
|
|
} else if (strncmp (buffer, ".\\\"", 3) == 0) {
|
|
|
|
/* Comment */
|
|
|
|
} else if (strcmp (buffer, ".TH") == 0) {
|
|
|
|
/* Title header */
|
2002-09-01 20:54:11 +04:00
|
|
|
} else if (strcmp (buffer, ".SM") == 0) {
|
|
|
|
/* Causes the text on the same line or the text on the
|
|
|
|
* next line to appear in a font that is one point
|
|
|
|
* size smaller than the default font. */
|
|
|
|
buffer = strtok (NULL, "");
|
|
|
|
if (buffer)
|
|
|
|
print_string (buffer);
|
2002-09-22 23:19:45 +04:00
|
|
|
} else if (handle_alt_font (buffer) == 1) {
|
|
|
|
return;
|
2002-07-31 02:57:01 +04:00
|
|
|
} else {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Other commands are ignored */
|
2002-09-01 14:45:35 +04:00
|
|
|
/* There is no memmove on some systems */
|
2002-09-06 00:47:12 +04:00
|
|
|
char warn_str[BUFFER_SIZE];
|
|
|
|
strcpy (warn_str, "Warning: unsupported command ");
|
|
|
|
strncat (warn_str, buffer, sizeof(warn_str));
|
|
|
|
print_error (warn_str);
|
2002-09-01 14:45:35 +04:00
|
|
|
return;
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-10-15 18:41:27 +04:00
|
|
|
static struct links {
|
|
|
|
char *linkname; /* Section name */
|
|
|
|
int line; /* Input line in ... */
|
|
|
|
const char *filename;
|
|
|
|
struct links *next;
|
|
|
|
} links, *current_link;
|
|
|
|
|
2002-07-31 02:57:01 +04:00
|
|
|
static void
|
|
|
|
handle_link (char *buffer)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
2002-07-31 02:57:01 +04:00
|
|
|
static char old[80];
|
1998-02-27 07:54:42 +03:00
|
|
|
int len;
|
|
|
|
|
2002-07-31 02:57:01 +04:00
|
|
|
switch (link_flag) {
|
1998-02-27 07:54:42 +03:00
|
|
|
case 1:
|
2001-08-16 05:36:41 +04:00
|
|
|
/* Old format link, not supported */
|
1998-02-27 07:54:42 +03:00
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
/* First part of new format link */
|
2002-03-21 16:01:20 +03:00
|
|
|
/* Bold text or italics text */
|
2002-07-31 02:57:01 +04:00
|
|
|
if (buffer[0] == '.' && (buffer[1] == 'I' || buffer[1] == 'B'))
|
|
|
|
for (buffer += 2; *buffer == ' ' || *buffer == '\t'; buffer++);
|
1998-02-27 07:54:42 +03:00
|
|
|
strcpy (old, buffer);
|
|
|
|
link_flag = 3;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
/* Second part of new format link */
|
2002-07-31 02:57:01 +04:00
|
|
|
if (buffer[0] == '.')
|
1998-02-27 07:54:42 +03:00
|
|
|
buffer++;
|
2002-07-31 02:57:01 +04:00
|
|
|
if (buffer[0] == '\\')
|
1998-02-27 07:54:42 +03:00
|
|
|
buffer++;
|
2002-07-31 02:57:01 +04:00
|
|
|
if (buffer[0] == '"')
|
1998-02-27 07:54:42 +03:00
|
|
|
buffer++;
|
|
|
|
len = strlen (buffer);
|
2002-07-31 02:57:01 +04:00
|
|
|
if (len && buffer[len - 1] == '"') {
|
|
|
|
buffer[--len] = 0;
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
2002-07-31 02:57:01 +04:00
|
|
|
printf_string ("%c%s%c%s%c\n", CHAR_LINK_START, old,
|
|
|
|
CHAR_LINK_POINTER, buffer, CHAR_LINK_END);
|
1998-02-27 07:54:42 +03:00
|
|
|
link_flag = 0;
|
2002-10-15 18:41:27 +04:00
|
|
|
/* Add to the linked list */
|
|
|
|
if (current_link) {
|
|
|
|
current_link->next = malloc (sizeof (links));
|
|
|
|
current_link = current_link->next;
|
|
|
|
current_link->next = NULL;
|
|
|
|
} else {
|
|
|
|
current_link = &links;
|
|
|
|
}
|
|
|
|
while (*buffer == ' ')
|
|
|
|
buffer++;
|
|
|
|
current_link->linkname = strdup (buffer);
|
|
|
|
current_link->filename = c_in;
|
|
|
|
current_link->line = in_row;
|
1998-02-27 07:54:42 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-07-31 02:57:01 +04:00
|
|
|
int
|
|
|
|
main (int argc, char **argv)
|
1998-02-27 07:54:42 +03:00
|
|
|
{
|
|
|
|
int len; /* Length of input line */
|
2002-08-20 20:13:13 +04:00
|
|
|
const char *c_man; /* Manual filename */
|
|
|
|
const char *c_tmpl; /* Template filename */
|
|
|
|
FILE *f_man; /* Manual file */
|
|
|
|
FILE *f_tmpl; /* Template file */
|
2002-08-27 08:27:26 +04:00
|
|
|
char buffer[BUFFER_SIZE]; /* Full input line */
|
2002-07-31 02:20:26 +04:00
|
|
|
char *node = NULL;
|
2002-08-27 08:27:26 +04:00
|
|
|
char *outfile_buffer; /* Large buffer to keep the output file */
|
|
|
|
long cont_start; /* Start of [Contents] */
|
|
|
|
long file_end; /* Length of the output file */
|
1998-02-27 07:54:42 +03:00
|
|
|
|
|
|
|
/* Validity check for arguments */
|
2002-09-22 21:03:28 +04:00
|
|
|
if (argc != 4) {
|
2002-07-31 02:57:01 +04:00
|
|
|
fprintf (stderr,
|
2002-09-22 21:03:28 +04:00
|
|
|
"Usage: man2hlp file.man template_file helpfile\n");
|
1998-02-27 07:54:42 +03:00
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
|
2002-09-22 21:03:28 +04:00
|
|
|
c_man = argv[1];
|
|
|
|
c_tmpl = argv[2];
|
|
|
|
c_out = argv[3];
|
2002-07-31 02:20:26 +04:00
|
|
|
|
2002-08-27 08:27:26 +04:00
|
|
|
/* First stage - process the manual, write to the output file */
|
|
|
|
f_man = fopen_check (c_man, "r");
|
|
|
|
f_out = fopen_check (c_out, "w");
|
2002-09-01 14:45:35 +04:00
|
|
|
c_in = c_man;
|
2002-07-31 02:20:26 +04:00
|
|
|
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Repeat for each input line */
|
2002-08-20 20:13:13 +04:00
|
|
|
while (!feof (f_man)) {
|
2002-08-27 08:27:26 +04:00
|
|
|
char *input_line; /* Input line without initial "\&" */
|
|
|
|
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Read a line */
|
2002-08-27 08:27:26 +04:00
|
|
|
if (!fgets (buffer, BUFFER_SIZE, f_man)) {
|
1998-02-27 07:54:42 +03:00
|
|
|
break;
|
|
|
|
}
|
2002-08-27 08:27:26 +04:00
|
|
|
|
|
|
|
if (buffer[0] == '\\' && buffer[1] == '&')
|
|
|
|
input_line = buffer + 2;
|
2001-08-16 05:36:41 +04:00
|
|
|
else
|
2002-08-27 08:27:26 +04:00
|
|
|
input_line = buffer;
|
|
|
|
|
2002-07-31 02:57:01 +04:00
|
|
|
in_row++;
|
2002-08-27 08:27:26 +04:00
|
|
|
len = strlen (input_line);
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Remove terminating newline */
|
2002-08-27 08:27:26 +04:00
|
|
|
if (input_line[len - 1] == '\n') {
|
2002-07-31 02:57:01 +04:00
|
|
|
len--;
|
2002-08-27 08:27:26 +04:00
|
|
|
input_line[len] = 0;
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
2002-08-27 08:27:26 +04:00
|
|
|
|
2002-07-31 02:57:01 +04:00
|
|
|
if (verbatim_flag) {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* Copy the line verbatim */
|
2002-08-27 08:27:26 +04:00
|
|
|
if (strcmp (input_line, ".fi") == 0) {
|
1998-02-27 07:54:42 +03:00
|
|
|
verbatim_flag = 0;
|
|
|
|
} else {
|
2002-08-27 08:27:26 +04:00
|
|
|
print_string (input_line);
|
1998-02-27 07:54:42 +03:00
|
|
|
newline ();
|
|
|
|
}
|
2002-07-31 02:57:01 +04:00
|
|
|
} else if (link_flag)
|
1998-02-27 07:54:42 +03:00
|
|
|
/* The line is a link */
|
2002-08-27 08:27:26 +04:00
|
|
|
handle_link (input_line);
|
1998-02-27 07:54:42 +03:00
|
|
|
else if (buffer[0] == '.')
|
|
|
|
/* The line is a roff command */
|
2002-08-27 08:27:26 +04:00
|
|
|
handle_command (input_line);
|
2002-07-31 02:57:01 +04:00
|
|
|
else {
|
1998-02-27 07:54:42 +03:00
|
|
|
/* A normal line, just output it */
|
2002-08-27 08:27:26 +04:00
|
|
|
print_string (input_line);
|
1998-02-27 07:54:42 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
newline ();
|
2002-08-27 08:27:26 +04:00
|
|
|
fclose_check (f_man);
|
|
|
|
/* First stage ends here, closing the manual */
|
2002-07-31 02:20:26 +04:00
|
|
|
|
2002-08-27 08:27:26 +04:00
|
|
|
/* Second stage - process the template file */
|
|
|
|
f_tmpl = fopen_check (c_tmpl, "r");
|
2002-09-01 14:45:35 +04:00
|
|
|
c_in = c_tmpl;
|
2002-08-20 20:13:13 +04:00
|
|
|
|
2002-07-31 02:20:26 +04:00
|
|
|
/* Repeat for each input line */
|
2002-10-15 18:41:27 +04:00
|
|
|
/* Read a line */
|
|
|
|
while (fgets (buffer, BUFFER_SIZE, f_tmpl)) {
|
2002-07-31 02:57:01 +04:00
|
|
|
if (node) {
|
2002-08-27 08:27:26 +04:00
|
|
|
if (*buffer && *buffer != '\n') {
|
|
|
|
cnode->lname = strdup (buffer);
|
2002-07-31 02:20:26 +04:00
|
|
|
node = strchr (cnode->lname, '\n');
|
|
|
|
if (node)
|
|
|
|
*node = 0;
|
|
|
|
}
|
|
|
|
node = NULL;
|
|
|
|
} else {
|
2002-07-31 02:57:01 +04:00
|
|
|
node = strchr (buffer, CHAR_NODE_END);
|
|
|
|
if (node && (node[1] == '[')) {
|
2002-10-15 18:41:27 +04:00
|
|
|
char *p = strchr (node, ']');
|
|
|
|
if (p) {
|
|
|
|
if (strncmp (node + 1, "[main]", 6) == 0) {
|
|
|
|
node = 0;
|
2002-07-31 02:57:01 +04:00
|
|
|
} else {
|
2002-10-15 18:41:27 +04:00
|
|
|
if (!cnode) {
|
|
|
|
cnode = &nodes;
|
|
|
|
} else {
|
|
|
|
cnode->next = malloc (sizeof (nodes));
|
|
|
|
cnode = cnode->next;
|
|
|
|
}
|
|
|
|
cnode->node = strdup (node + 2);
|
|
|
|
cnode->node[p - node - 2] = 0;
|
|
|
|
cnode->lname = NULL;
|
|
|
|
cnode->next = NULL;
|
2002-07-31 02:57:01 +04:00
|
|
|
}
|
2002-10-15 18:41:27 +04:00
|
|
|
} else
|
|
|
|
node = NULL;
|
2002-07-31 02:57:01 +04:00
|
|
|
} else
|
|
|
|
node = NULL;
|
2002-07-31 02:20:26 +04:00
|
|
|
}
|
2002-08-20 20:13:13 +04:00
|
|
|
fputs (buffer, f_out);
|
2002-07-31 02:20:26 +04:00
|
|
|
}
|
|
|
|
|
2002-08-20 20:13:13 +04:00
|
|
|
cont_start = ftell (f_out);
|
2002-08-27 08:27:26 +04:00
|
|
|
if (topics)
|
|
|
|
fprintf (f_out, "\004[Contents]\n%s\n\n", topics);
|
2002-08-25 22:40:33 +04:00
|
|
|
else
|
|
|
|
fprintf (f_out, "\004[Contents]\n");
|
2002-07-31 02:20:26 +04:00
|
|
|
|
2002-10-15 18:41:27 +04:00
|
|
|
for (current_link = &links; current_link && current_link->linkname;) {
|
|
|
|
int found = 0;
|
|
|
|
struct links *next = current_link->next;
|
|
|
|
|
|
|
|
if (strcmp (current_link->linkname, "Contents") == 0) {
|
|
|
|
found = 1;
|
|
|
|
} else {
|
|
|
|
for (cnode = &nodes; cnode && cnode->node; cnode = cnode->next) {
|
|
|
|
if (strcmp (cnode->node, current_link->linkname) == 0) {
|
|
|
|
found = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!found) {
|
2002-10-15 19:38:15 +04:00
|
|
|
g_snprintf (buffer, sizeof (buffer), "Stale link \"%s\"",
|
|
|
|
current_link->linkname);
|
2002-10-15 18:41:27 +04:00
|
|
|
c_in = current_link->filename;
|
|
|
|
in_row = current_link->line;
|
|
|
|
print_error (buffer);
|
|
|
|
}
|
|
|
|
free (current_link->linkname);
|
|
|
|
if (current_link != &links)
|
|
|
|
free (current_link);
|
|
|
|
current_link = next;
|
|
|
|
}
|
|
|
|
|
2002-07-31 04:08:51 +04:00
|
|
|
for (cnode = &nodes; cnode && cnode->node;) {
|
2002-07-31 02:20:26 +04:00
|
|
|
char *node = cnode->node;
|
2002-07-31 04:08:51 +04:00
|
|
|
struct node *next = cnode->next;
|
|
|
|
|
2002-07-31 02:20:26 +04:00
|
|
|
if (*node)
|
2002-10-15 18:41:27 +04:00
|
|
|
fprintf (f_out, " %*s\001 %s \002%s\003", cnode->heading_level,
|
|
|
|
"", cnode->lname ? cnode->lname : node, node);
|
2002-08-20 20:13:13 +04:00
|
|
|
fprintf (f_out, "\n");
|
2002-07-31 02:20:26 +04:00
|
|
|
|
|
|
|
free (cnode->node);
|
|
|
|
if (cnode->lname)
|
|
|
|
free (cnode->lname);
|
|
|
|
if (cnode != &nodes)
|
|
|
|
free (cnode);
|
2002-07-31 04:08:51 +04:00
|
|
|
cnode = next;
|
2002-07-31 02:20:26 +04:00
|
|
|
}
|
|
|
|
|
2002-08-20 20:13:13 +04:00
|
|
|
file_end = ftell (f_out);
|
|
|
|
if (file_end <= 0) {
|
|
|
|
perror (c_out);
|
2002-08-20 20:14:48 +04:00
|
|
|
return 1;
|
2002-08-20 20:13:13 +04:00
|
|
|
}
|
2002-07-31 02:20:26 +04:00
|
|
|
|
2002-08-27 08:27:26 +04:00
|
|
|
fclose_check (f_out);
|
|
|
|
fclose_check (f_tmpl);
|
|
|
|
/* Second stage ends here, closing all files, note the end of output */
|
2002-07-31 02:57:01 +04:00
|
|
|
|
2002-08-27 08:27:26 +04:00
|
|
|
/*
|
|
|
|
* Third stage - swap two parts of the output file.
|
|
|
|
* First, open the output file for reading and load it into the memory.
|
|
|
|
*/
|
|
|
|
f_out = fopen_check (c_out, "r");
|
2002-07-31 02:57:01 +04:00
|
|
|
|
2002-08-27 08:27:26 +04:00
|
|
|
outfile_buffer = malloc (file_end);
|
|
|
|
if (!outfile_buffer)
|
2002-07-31 02:57:01 +04:00
|
|
|
return 1;
|
|
|
|
|
2002-08-27 08:27:26 +04:00
|
|
|
if (persistent_fread (outfile_buffer, file_end, f_out) < 0) {
|
2002-08-20 20:13:13 +04:00
|
|
|
perror (c_out);
|
2002-07-31 02:57:01 +04:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2002-08-27 08:27:26 +04:00
|
|
|
fclose_check (f_out);
|
|
|
|
/* Now the output file is in the memory */
|
2002-07-31 02:57:01 +04:00
|
|
|
|
2002-08-27 08:27:26 +04:00
|
|
|
/* Again open output file for writing */
|
|
|
|
f_out = fopen_check (c_out, "w");
|
|
|
|
|
|
|
|
/* Write part after the "Contents" node */
|
2002-08-16 20:48:27 +04:00
|
|
|
if (persistent_fwrite
|
2002-08-27 08:27:26 +04:00
|
|
|
(outfile_buffer + cont_start, file_end - cont_start, f_out) < 0) {
|
2002-08-20 20:13:13 +04:00
|
|
|
perror (c_out);
|
2002-07-31 02:57:01 +04:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2002-08-27 08:27:26 +04:00
|
|
|
/* Write part before the "Contents" node */
|
|
|
|
if (persistent_fwrite (outfile_buffer, cont_start, f_out) < 0) {
|
2002-08-20 20:13:13 +04:00
|
|
|
perror (c_out);
|
2002-07-31 02:57:01 +04:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2002-08-27 08:27:26 +04:00
|
|
|
free (outfile_buffer);
|
|
|
|
fclose_check (f_out);
|
|
|
|
/* Closing everything */
|
2002-07-31 02:20:26 +04:00
|
|
|
|
1998-02-27 07:54:42 +03:00
|
|
|
return 0;
|
|
|
|
}
|