Logo Search packages:      
Sourcecode: w3mmee version File versions  Download package

etc.c

#include "fm.h"
#include <pwd.h>
#include "myctype.h"
#include "html.h"
#include "local.h"
#include "terms.h"

#ifdef HAVE_GETCWD /* ??? ukai */
#include <unistd.h>
#include <sys/param.h>
#endif                        /* HAVE_GETCWD */

#include <sys/types.h>
#include <time.h>
#if defined(HAVE_WAITPID) || defined(HAVE_WAIT3)
#include <sys/wait.h>
#endif
#include <signal.h>
#include <errno.h>

#ifdef      __WATT32__
#define     read(a,b,c) read_s(a,b,c)
#define     close(x)    close_s(x)
#endif                        /* __WATT32__ */

struct auth_pass {
    int is_proxy;
    Str host;
    int port;
    Str file;
    Str realm;
    Str uname;
    Str pwd;
    struct auth_pass *next;
};

struct auth_pass *passwords = NULL;

int
columnSkip(Buffer *buf, int offset)
{
    int i, maxColumn, column, nlines;
    Line *l;
    BufferView *v;

    if (!(v = buf->view))
      return 0;
    column = buf->currentColumn + offset;
    nlines = v->height;
    maxColumn = 0;
    for (i = 0, l = buf->topLine;
       i < nlines && l != NULL;
       i++, l = l->next) {
      if (l->width < 0)
          l->width = COLPOS(l, l->len);
      if (l->width - 1 > maxColumn)
          maxColumn = l->width - 1;
    }
    maxColumn -= v->width - buf->rmargin - buf->lmargin - 1;
    if (column < maxColumn)
      maxColumn = column;
    if (maxColumn < 0)
      maxColumn = 0;

    if (buf->currentColumn == maxColumn)
      return 0;
    buf->currentColumn = maxColumn;
    return 1;
}

int
columnPos(Line *line, int column)
{
    int i;
#ifdef MANY_CHARSET
    size_t cb, ce;
#endif

    for (i = 1; i < line->len; i++) {
      if (COLPOS(line, i) > column) {
#ifdef MANY_CHARSET
          break;
#else
#ifdef JP_CHARSET
          if (CharType(line->propBuf[i - 1]) == PC_KANJI2)
            return i - 2;
#endif
          return i - 1;
#endif
      }
    }
#ifdef MANY_CHARSET
    cb = i - 1;
    ce = line->len;
    mb_mem_to_wchar(line->lineBuf, &cb, &ce);
    return (int)cb;
#else
    return i - 1;
#endif
}

Line *
lineSkip(Buffer *buf, Line *line, int offset, int last)
{
    int i;
    Line *l;

    l = currentLineSkip(buf, line, offset, last);
    for (i = last - (buf->lastLine->linenumber - l->linenumber);
      i > 0 && l->prev != NULL;
      i--, l = l->prev);
    return l;
}

Line *
currentLineSkip(Buffer *buf, Line *line, int offset, int last)
{
    int i, n;
    Line *l = line;

    n = line->linenumber + offset + last + 1;

    if (buf->lastLine->linenumber < n &&
      buf->async_buf && buf->async_buf->rfd >= 0 &&
      !is_recorded_read_fd(buf->async_buf->rfd) &&
      is_hook_recorded_read_fd(buf->async_buf->rfd))
      record_read_fd(buf->async_buf->rfd, NULL, NULL, NULL);

    if (offset > 0)
      for (i = 0; i < offset && l->next != NULL; i++, l = l->next);
    else if (offset < 0)
      for (i = 0; i < -offset && l->prev != NULL; i++, l = l->prev);
    return l;
}

#define MAX_CMD_LEN 128

int
gethtmlcmd(char **s)
{
    char cmdstr[MAX_CMD_LEN];
    char *p = cmdstr;
    char *save = *s;
    int cmd;
    TagInfo *taginfo;
    extern btri_string_tab_t tagtable[];

    (*s)++;
    /* first character */
    if (IS_ALNUM(**s) || **s == '_' || **s == '/')
      *(p++) = tolower(*((*s)++));
    else
      return HTML_UNKNOWN;
    if (p[-1] == '/')
      SKIP_BLANKS(*s);
    while ((IS_ALNUM(**s) || **s == '_') &&
         p - cmdstr < MAX_CMD_LEN) {
      *(p++) = tolower(*((*s)++));
    }
    if (p - cmdstr == MAX_CMD_LEN) {
      /* buffer overflow: perhaps caused by bad HTML source */
      *s = save + 1;
      return HTML_UNKNOWN;
    }
    *p = '\0';

    if (btri_fast_ci_search_mem(cmdstr, p - cmdstr, tagtable, (void **)&taginfo) != bt_failure)
      cmd = taginfo - TagMAP;
    else
      cmd = HTML_UNKNOWN;
    while (**s && **s != '>')
      (*s)++;
    if (**s == '>')
      (*s)++;
    return cmd;
}

#define CHECKTYPE_NORMAL (0)
#define CHECKTYPE_CONTINUE (1)
#define CHECKTYPE_PENDING (2)

#ifdef USE_ANSI_COLOR
static int
parse_ansi_color(char **str, char *endp, Lineprop *effect, Linecolor *color)
{
    char *p = *str, *q;
    Lineprop e = *effect;
    Linecolor c = *color;
    int i;

    if (endp - p <= 2)
      return CHECKTYPE_PENDING;
    if (*(p+1) != '[')
      return CHECKTYPE_NORMAL;
    p += 2;
    for (q = p; q < endp && (IS_DIGIT(*q) || *q == ';'); q++)
      ;
    if (q >= endp)
      return CHECKTYPE_PENDING;
    if (q >= endp || *q != 'm')
      return CHECKTYPE_NORMAL;
    *str = q + 1;
    while (1) {
      if (*p == 'm') {
          e = PE_NORMAL;
          c = 0;
          break;
      }
      if (IS_DIGIT(*p)) {
          q = p;
          for (p++; IS_DIGIT(*p); p++)
            ;
          i = atoi(allocStr(q, p - q));
          switch (i) {
          case 0:
            e = PE_NORMAL;
            c = 0;
            break;
          case 1:
          case 5:
            e = PE_BOLD;
            break;
          case 4:
            e = PE_UNDER;
            break;
          case 7:
            e = PE_STAND;
            break;
          case 100:     /* for EWS4800 kterm */
            c = 0;
            break;
          case 39:
            c &= 0xf0;
            break;
          case 49:
            c &= 0x0f;
            break;
          default:
            if (i >= 30 && i <= 37)
                c = (c & 0xf0) | (i - 30) | 0x08;
            else if (i >= 40 && i <= 47)
                c = (c & 0x0f) | ((i - 40) << 4) | 0x80;
            break;
          }
          if (*p == 'm')
            break;
      } else {
          e = PE_NORMAL;
          c = 0;
          break;
      }
      p++;  /* *p == ';' */
    }
    *effect = e;
    *color = c;
    return CHECKTYPE_CONTINUE;
}
#endif

static int
parse_backspace(char **p_str, char *endp, int eol_p,
            char *begp, char **p_dest, Lineprop **p_prop
#ifdef USE_ANSI_COLOR
            , Linecolor **p_color
#endif
            )
{
  char *str, *dest;
  Lineprop *prop;
#ifdef USE_ANSI_COLOR
  Linecolor *color;
#endif
#ifdef MANY_CHARSET
  size_t cb, ce, ce1;
  mb_wchar_t wc;
#endif

  str = *p_str;
  dest = *p_dest;
  prop = *p_prop;
#ifdef USE_ANSI_COLOR
  color = *p_color;
#endif

  if (str + 1 < endp) {
    if (*(str + 1) == '\b') {
      if (str + 2 < endp) {
      if (*(str + 2) == '_') {
        if (str + 3 < endp) {
          if (*(str + 3) == '_') {
            if (dest > begp) {
#ifdef MANY_CHARSET
            ce = dest - begp;
            cb = ce - 1;
            mb_mem_to_wchar(begp, &cb, &ce);

            for (ce -= cb ; ce > 0 ; --ce)
              *(prop - ce) |= PE_UNDER;
#else
#ifdef JP_CHARSET
            if (dest - begp >= 2 && CharType(*(prop - 2)) == PC_KANJI1)
              *(prop - 2) |= PE_UNDER;
#endif
            *(prop - 1) |= PE_UNDER;
#endif
            *p_str = str + 4;
            return CHECKTYPE_CONTINUE;
            }
            else {
            *p_str = str += 2;
            return CHECKTYPE_CONTINUE;
            }
          }
        }
        else if (!eol_p)
          return CHECKTYPE_PENDING;
      }
      }
      else if (!eol_p)
      return CHECKTYPE_PENDING;

      if (dest > begp) {
#ifdef MANY_CHARSET
      ce1 = endp - (str + 2);
      cb = 0;
      wc = mb_mem_to_wchar(str + 2, &cb, &ce1);

      if (ce1 == endp - (str + 2) && !eol_p)
        return CHECKTYPE_PENDING;

      ce = dest - begp;
      cb = ce - 1;

      if (mb_mem_to_wchar(begp, &cb, &ce) == wc) {
        for (ce -= cb ; ce > 0 ; --ce)
          *(prop - ce) |= PE_BOLD;

        *p_str = str + 2 + ce1;
      }
      else {
        *p_dest = dest - ce;
        *p_prop = prop - ce;
#ifdef USE_ANSI_COLOR
        if (color) *p_color = color - ce;
#endif
        ++str;
      }

      return CHECKTYPE_CONTINUE;
#else
#ifdef JP_CHARSET
      if (dest - begp >= 2 && CharType(*(prop - 2)) == PC_KANJI1) {
        if (str + 3 >= endp && !eol_p)
          return TRUE;

        if (*(str + 2) == *(dest - 2) && *(str + 3) == *(dest - 1)) {
          *(prop - 2) |= PE_BOLD;
          *(prop - 1) |= PE_BOLD;
          *p_str = str + 4;
        }
        else {
          *p_dest = dest - 2;
          *p_prop = prop - 2;
#ifdef USE_ANSI_COLOR
          if (color) *p_color = color - 2;
#endif
          *p_str = str + 1;
        }

        return FALSE;
      }
      else
#endif
        {
          if (*(str + 2) == *(dest - 1)) {
            *(prop - 1) |= PE_BOLD;
            *p_str = str + 3;
          }
          else {
            *p_dest = dest - 1;
            *p_prop = prop - 1;
#ifdef USE_ANSI_COLOR
            if (color) *p_color = color - 1;
#endif
            *p_str = str + 1;
          }

          return CHECKTYPE_CONTINUE;
        }
#endif
      }
      else {
      *p_str = str + 1;
      return CHECKTYPE_CONTINUE;
      }
    }
    else if (*(str + 1) == '_') {
      if (dest > begp) {
#ifdef MANY_CHARSET
      ce = dest - begp;
      cb = ce - 1;
      mb_mem_to_wchar(begp, &cb, &ce);

      for (ce -= cb ; ce > 0 ; --ce)
        *(prop - ce) |= PE_UNDER;
#else
#ifdef JP_CHARSET
      if (dest - begp >= 2 && CharType(*(prop - 2)) == PC_KANJI1)
        *(prop - 2) |= PE_UNDER;
#endif
      *(prop - 1) |= PE_UNDER;
#endif
      *p_str = str + 3;
      }
      else
      *p_str = str + 1;

      return CHECKTYPE_CONTINUE;
    }
  }
  else if (!eol_p)
    return CHECKTYPE_PENDING;

  if (dest > begp) {
#ifdef MANY_CHARSET
    ce1 = endp - (str + 1);
    cb = 0;
    wc = mb_mem_to_wchar(str + 1, &cb, &ce1);

    if (ce1 == endp - (str + 1) && !eol_p)
      return CHECKTYPE_PENDING;

    ce = dest - begp;
    cb = ce - 1;

    if (mb_mem_to_wchar(begp, &cb, &ce) == wc) {
      for (ce -= cb ; ce > 0 ; --ce)
      *(prop - ce) |= PE_BOLD;

      *p_str = str + 1 + ce1;
    }
    else {
      *p_dest = dest - ce;
      *p_prop = prop - ce;
#ifdef USE_ANSI_COLOR
      if (color) *p_color = color - ce;
#endif
      *p_str = str + 1;
    }

    return CHECKTYPE_CONTINUE;
#else
#ifdef JP_CHARSET
    if (dest - begp >= 2 && CharType(*(prop - 2)) == PC_KANJI1) {
      if (str + 2 >= endp && !eol_p)
      return -1;

      if (*(str + 1) == *(dest - 2) && *(str + 2) == *(dest - 1)) {
      *(prop - 2) |= PE_BOLD;
      *(prop - 1) |= PE_BOLD;
      *p_str = str + 3;
      }
      else {
      *p_dest = dest - 2;
      *p_prop = prop - 2;
#ifdef USE_ANSI_COLOR
      if (color) *p_color = color - 2;
#endif
      *p_str = str + 1;
      }

      return CHECKTYPE_CONTINUE;
    }
    else
#endif
      {
      if (*(str + 1) == *(dest - 1)) {
        *(prop - 1) |= PE_BOLD;
        *p_str = str + 2;
      }
      else {
        *p_dest = dest - 1;
        *p_prop = prop - 1;
#ifdef USE_ANSI_COLOR
        if (color) *p_color = color - 1;
#endif
        *p_str = str + 1;
      }

      return CHECKTYPE_CONTINUE;
      }
#endif
  }

  *p_str = str + 1;
  return CHECKTYPE_CONTINUE;
}

static int
parse_underscore(char **p_str, char *endp, int eol_p, Lineprop *p_effect)
{
  char *str;

  str = *p_str;

  if (str + 1 < endp) {
    if (*(str + 1) == '\b') {
      *p_effect |= PE_UNDER;
      *p_str = str + 2;
      return CHECKTYPE_CONTINUE;
    }
    else if (*(str + 1) == '_') {
      if (str + 2 < endp) {
      if (*(str + 2) == '\b') {
        if (str + 3 < endp) {
          if (*(str + 3) == '\b') {
            *p_effect |= PE_UNDER;
            *p_str = str + 4;
            return CHECKTYPE_CONTINUE;
          }
        }
        else if (!eol_p)
          return CHECKTYPE_PENDING;
      }
      }
      else if (!eol_p)
      return CHECKTYPE_PENDING;
    }
  }
  else if (!eol_p)
    return CHECKTYPE_PENDING;

  return CHECKTYPE_NORMAL;
}

/* 
 * Check character type
 */

int
checkTypeCat(CheckTypeEnv *p)
{
  Lineprop *prop;
  char *begp, *str, *endp, *dest;
#ifdef USE_ANSI_COLOR
  Linecolor *color = NULL;
#endif
#ifdef MANY_CHARSET
  int cn;
  mb_wchar_t wc;
#endif
  int mode, res, col;

  str = p->from;
  endp = p->from_end;
  dest = begp = p->to;
  prop = p->to_prop;
#ifdef USE_ANSI_COLOR
  color = p->to_color;
#endif

  if (endp - str > p->to_size - (dest - p->to_beg)) {
    int off;

    off = dest - p->to_beg;

#ifdef USE_ANSI_COLOR
    if (color) {
      new_objv(&p->to_size, off + (endp - str),
             &p->to_beg, sizeof(p->to_beg[0]), TRUE,
             &p->to_prop_beg, sizeof(p->to_prop_beg[0]), TRUE,
             &p->to_color_beg, sizeof(p->to_color_beg[0]), TRUE,
             NULL);
      p->to_color = color = &p->to_color_beg[off];
    }
    else
#endif
      {
      new_objv(&p->to_size, off + (endp - str),
             &p->to_beg, sizeof(p->to_beg[0]), TRUE,
             &p->to_prop_beg, sizeof(p->to_prop_beg[0]), TRUE,
             NULL);
      }

    p->to = dest = begp = &p->to_beg[off];
    p->to_prop = prop = &p->to_prop_beg[off];
  }

  while (str < endp)
    switch (*str) {
    case '\0':
      *str = ' ';
      goto normal;
    case '\b':
      if (ShowEffect) {
      if (parse_backspace(&str, endp, p->eol_p, begp, &dest, &prop
#ifdef USE_ANSI_COLOR
                      , &color
#endif
                      ) == CHECKTYPE_PENDING)
        goto pending;
      else
        continue;
      }

      goto normal;
    case '\t':
      col = (p->col / Tabstop + 1) * Tabstop;

      if (p->width && col > p->width) {
      res = CHECKTYPE_EOL;
      goto end;
      }

      p->col = col;
      *dest++ = '\t';
      *prop++ = (p->effect
#ifdef USE_ANSI_COLOR
             | p->ceffect
#endif
             | PC_CTRL);
#ifdef USE_ANSI_COLOR
      if (color) *color++ = p->cmode;
#endif
      p->effect = PE_NORMAL;
      ++str;
      continue;
    case '\r':
      if (str + 1 < endp) {
      if (*(str + 1) == '\n')
        ++str;

      goto newline;
      }
      else if (!p->eol_p)
      goto pending;
    newline:
    case '\n':
      res = CHECKTYPE_EOL | CHECKTYPE_NL;
      ++str;
      goto end;
#ifdef USE_ANSI_COLOR
    case '\e':
      if (p->use_color) {
      char *q;

      q = str;

      switch (parse_ansi_color(&str, endp, &p->ceffect, &p->cmode)) {
      default: /* case CHECKTYPE_NORMAL: */
        break;
      case CHECKTYPE_CONTINUE:
        if (p->cmode && !color) {
          p->to_color_beg = NewAtom_N(Linecolor, p->to_size);
          memset(p->to_color_beg, 0, sizeof(p->to_color_beg[0]) * p->to_size);
          color = p->to_color_beg + (dest - p->to_beg);
          p->to_color = p->to_color_beg + (p->to - p->to_beg);
        }

        continue;
      case CHECKTYPE_PENDING:
        goto pending;
      }

      str = q;
      }

      goto normal;
#endif
    case '_':
      if (ShowEffect)
      switch (parse_underscore(&str, endp, p->eol_p, &p->effect)) {
      default: /* case CHECKTYPE_NORMAL: */
        break;
      case CHECKTYPE_CONTINUE:
        continue;
      case CHECKTYPE_PENDING:
        goto pending;
      }
    normal:
    default:
      mode = get_mctype(str);
#ifdef MANY_CHARSET
      if ((cn = mb_mem_to_wchar_internal(str, endp - str, wc)) < 0) {
      if (endp - str < MB_MBC_LEN_MAX && !p->eol_p)
        goto pending;
      else
        cn = 1;
      }

      if (p->width) {
      int cw;

      cw = ttyfix_wchar_width(wc);

      if (p->col + cw > p->width) {
        res = CHECKTYPE_EOL;
        goto end;
      }

      p->col += cw;
      }

      switch (cn) {
      case MB_MBC_LEN_MAX:
#ifdef USE_ANSI_COLOR
        if (color) {
            int i;

            for (i = 0 ; i < MB_MBC_LEN_MAX - 1 ; ++i)
              *color++ = p->cmode;
        }
#endif
        for (; cn > 1 ; --cn) {
            *prop++ = (p->effect | mode);
            *dest++ = *str++;
        }
      default:
#ifdef USE_ANSI_COLOR
        if (color)
            *color++ = p->cmode;
#endif
        *prop++ = (p->effect | mode);
        *dest++ = *str++;
        break;
      }
#else
#ifdef JP_CHARSET
      if (mode == PC_KANJI) {
      if (str + 1 >= endp && !p->eol_p)
        goto pending;

      if (p->width) {
        if (p->col + 2 > p->width) {
          res = CHECKTYPE_EOL;
          goto end;
        }

        p->col += 2;
      }

      *prop++ = (p->effect | PC_KANJI1);
      *prop++ = (p->effect | PC_KANJI2);
      *dest++ = *str++;
      *dest++ = *str++;

#ifdef USE_ANSI_COLOR
      if (color) {
        *color++ = p->cmode;
        *color++ = p->cmode;
      }
#endif
      }
      else
#endif                        /* JP_CHARSET */
      {
        if (p->width) {
          if (p->col + 1 > p->width) {
            res = CHECKTYPE_EOL;
            goto end;
          }

          ++(p->col);
        }

        *prop++ = (p->effect | mode);
        *dest++ = *str++;

#ifdef USE_ANSI_COLOR
        if (color)
          *color++ = p->cmode;
#endif
      }
#endif
      p->effect = PE_NORMAL;
    }

  if (p->eol_p) {
    res = CHECKTYPE_EOL | CHECKTYPE_NL;
    goto end;
  }
pending:
  res = 0;
end:
  p->from = str;
  p->to = dest;
  p->to_prop = prop;
#ifdef USE_ANSI_COLOR
  p->to_color = color;
#endif
  return res;
}

int
calcPosition(char *l, Lineprop *pr, int len, int pos, int bpos, int mode)
{
    static int *realColumn;
    static int realColumnSize;
    static char *prevl = NULL;
    int i, j;
#ifdef MANY_CHARSET
    int ce;
    mb_wchar_t wc;
#endif

    if (l == NULL || len == 0)
      return bpos;
    if (l == prevl && mode == CP_AUTO) {
        if (pos <= len)
          return realColumn[pos];
    }
    prevl = l;
    j = bpos;
    for (i = 0 ;; ++i) {
      NEW_OBJV1(&realColumnSize, i, &realColumn, sizeof(realColumn[0]), TRUE);
      realColumn[i] = j;
        if (i == len)
            break;
      if (l[i] == '\t' && pr[i] == PC_CTRL)
          j = (j + Tabstop) / Tabstop * Tabstop;
#if !defined(KANJI_SYMBOLS) && !defined(MANY_CHARSET)
      else if (pr[i] & PC_RULE)
          ++j;
#endif
      else if (IS_UNPRINTABLE_ASCII(l[i], pr[i]))
          j += 4;
      else if (IS_UNPRINTABLE_CONTROL(l[i], pr[i]))
          j += 2;
#ifdef MANY_CHARSET
      else {
          if ((ce = mb_mem_to_wchar_internal(&l[i], len - i, wc)) < 0)
            ce = 1;
          NEW_OBJV1(&realColumnSize, i + ce, &realColumn, sizeof(realColumn[0]), TRUE);
          switch (ce) {
          case MB_MBC_LEN_MAX:
            for (; ce > 1 ; --ce)
                realColumn[++i] = j;
          default:
            break;
          }
          j += ttyfix_wchar_width(wc);
      }
#else
      else
          j++;
#endif
    }
    if (pos >= i)
      return j;
    return realColumn[pos];
}

char *
lastFileName(char *path)
{
    char *p, *q;

    p = q = path;
    while (*p != '\0') {
      if (*p == '/')
          q = p + 1;
      p++;
    }

    return allocStr(q, -1);
}

#ifdef USE_INCLUDED_SRAND48
static unsigned long R1 = 0x1234abcd;
static unsigned long R2 = 0x330e;
#define A1 0x5deec
#define A2 0xe66d
#define C 0xb

void
srand48(long seed)
{
    R1 = (unsigned long)seed;
    R2 = 0x330e;
}

long
lrand48(void)
{
    R1 = (A1 * R1 << 16) + A1 * R2 + A2 * R1 + ((A2 * R2 + C) >> 16);
    R2 = (A2 * R2 + C) & 0xffff;
    return (long)(R1 >> 1);
}
#endif

char *
mybasename(char *s)
{
    char *p = s;
    while (*p)
      p++;
    while (s <= p && *p != '/')
      p--;
    if (*p == '/')
      p++;
    else
      p = s;
    return allocStr(p, -1);
}

char *
mydirname(char *s)
{
    char *p = s;
    while (*p)
      p++;
    if (s != p)
      p--;
    while (s != p && *p == '/')
      p--;
    while (s != p && *p != '/')
      p--;
    if (*p != '/')
      return ".";
    while (s != p && *p == '/')
      p--;
    return allocStr(s, strlen(s) - strlen(p) + 1);
}

#ifndef HAVE_STRERROR
char *
strerror(int errno)
{
    extern char *sys_errlist[];
    return sys_errlist[errno];
}
#endif                        /* not HAVE_STRERROR */

#ifndef HAVE_SYS_ERRLIST
char **sys_errlist;

prepare_sys_errlist()
{
    int i, n;

    i = 1;
    while (strerror(i) != NULL)
      i++;
    n = i;
    sys_errlist = New_N(char *, n);
    sys_errlist[0] = "";
    for (i = 1; i < n; i++)
      sys_errlist[i] = strerror(i);
}
#endif                        /* not HAVE_SYS_ERRLIST */

int
next_status(char c, int *status)
{
    switch (*status) {
    case R_ST_NORMAL:
      if (c == '<') {
          *status = R_ST_TAG0;
          return 0;
      }
      else if (c == '&') {
          *status = R_ST_AMP;
          return 1;
      }
      else
          return 1;
      break;
    case R_ST_TAG0:
      if (c == '!') {
          *status = R_ST_CMNT1;
          return 0;
      }
      *status = R_ST_TAG;
      /* continues to next case */
    case R_ST_TAG:
      if (c == '>')
          *status = R_ST_NORMAL;
      else if (c == '=')
          *status = R_ST_EQL;
      return 0;
    case R_ST_EQL:
      if (c == '"')
          *status = R_ST_DQUOTE;
      else if (c == '\'')
          *status = R_ST_QUOTE;
      else if (IS_SPACE(c))
          *status = R_ST_EQL;
      else if (c == '>')
          *status = R_ST_NORMAL;
      else
          *status = R_ST_TAG;
      return 0;
    case R_ST_QUOTE:
      if (c == '\'')
          *status = R_ST_TAG;
      return 0;
    case R_ST_DQUOTE:
      if (c == '"')
          *status = R_ST_TAG;
      return 0;
    case R_ST_AMP:
      if (c == ';') {
          *status = R_ST_NORMAL;
          return 0;
      }
      else if (c != '#' && !IS_ALNUM(c) && c != '_') {
          /* something's wrong! */
          *status = R_ST_NORMAL;
          return 0;
      }
      else
          return 0;
    case R_ST_CMNT1:
      switch (c) {
      case '-':
          *status = R_ST_CMNT2;
          break;
      case '>':
          *status = R_ST_NORMAL;
          break;
      default:
          *status = R_ST_IRRTAG;
      }
      return 0;
    case R_ST_CMNT2:
      switch (c) {
      case '-':
          *status = R_ST_CMNT;
          break;
      case '>':
          *status = R_ST_NORMAL;
          break;
      default:
          *status = R_ST_IRRTAG;
      }
      return 0;
    case R_ST_CMNT:
      if (c == '-')
          *status = R_ST_NCMNT1;
      return 0;
    case R_ST_NCMNT1:
      if (c == '-')
          *status = R_ST_NCMNT2;
      else
          *status = R_ST_CMNT;
      return 0;
    case R_ST_NCMNT2:
      switch (c) {
      case '>':
          *status = R_ST_NORMAL;
          break;
      case '-':
          *status = R_ST_NCMNT2;
          break;
      default:
          if (!c || IS_SPACE(c))
            *status = R_ST_NCMNT3;
          else
            *status = R_ST_CMNT;
          break;
      }
      break;
    case R_ST_NCMNT3:
      switch (c) {
      case '>':
          *status = R_ST_NORMAL;
          break;
      case '-':
          *status = R_ST_NCMNT1;
          break;
      default:
          if (!c || IS_SPACE(c))
            *status = R_ST_NCMNT3;
          else
            *status = R_ST_CMNT;
          break;
      }
      return 0;
    case R_ST_IRRTAG:
      if (c == '>')
          *status = R_ST_NORMAL;
      return 0;
    }
    /* notreached */
    return 0;
}

int
read_token(Str buf, char **instr, char *instr_end, int *status, int flag)
{
    char *p;
    int prev_status;

    if (!(flag & RT_APPEND))
      Strclear(buf);
    if (*instr >= instr_end)
      return 0;
    for (p = *instr; p < instr_end ; p++) {
      if (*p == '\r') {
          if (p + 1 < instr_end) {
            if (*(p + 1) == '\n')
                ++p;
          }
          else if (!(flag & RT_EOL))
            break;
      }
      prev_status = *status;
      next_status(*p, status);
      switch (*status) {
      case R_ST_NORMAL:
          if (prev_status == R_ST_AMP && *p != ';') {
            p--;
            break;
          }
          if (prev_status == R_ST_NCMNT2 || prev_status == R_ST_NCMNT3 ||
            prev_status == R_ST_IRRTAG || prev_status == R_ST_CMNT1) {
            if (prev_status == R_ST_CMNT1 && !(flag & RT_APPEND))
                Strclear(buf);
            p++;
            goto proc_end;
          }
          if (!(flag & RT_PRE) && (!*p || IS_SPACE(*p)))
            Strcat_char(buf, ' ');
          else {
            Strcat_char(buf, *p);
            if (*p == '\n') {
                ++p;
                goto proc_end;
            }
          }
          if (ST_IS_REAL_TAG(prev_status)) {
            *instr = p + 1;
            if (buf->length < 2 ||
                buf->ptr[buf->length - 2] != '<' ||
                buf->ptr[buf->length - 1] != '>')
                return 1;
            Strshrink(buf, 2);
          }
          break;
      case R_ST_TAG0:
          if (prev_status == R_ST_NORMAL && p != *instr) {
            *instr = p;
            *status = prev_status;
            return 1;
          }
      case R_ST_TAG:
          if (*status == R_ST_TAG0 && p + 1 < instr_end &&
            !REALLY_THE_BEGINNING_OF_A_TAG(p)) {
            /* it seems that this '<' is not a beginning of a tag */
            Strcat_charp(buf, "&lt;");
            *status = R_ST_NORMAL;
            Strcat_char(buf, *p);
          }
          else if (!(flag & RT_OLDPRE) && (!*p || IS_SPACE(*p))) {
            if (!buf->length || (buf->ptr[buf->length - 1] && !IS_SPACE(buf->ptr[buf->length - 1])))
                Strcat_char(buf, ' ');
          }
          else
            Strcat_char(buf, *p);
          break;
      case R_ST_EQL:
          if ((flag & RT_OLDPRE) || (*p && !IS_SPACE(*p)))
            goto literal_cat;
          if (!buf->length || (buf->ptr[buf->length - 1] && !IS_SPACE(buf->ptr[buf->length - 1])))
            Strcat_char(buf, ' ');
          break;
      case R_ST_QUOTE:
      case R_ST_DQUOTE:
          if ((flag & RT_OLDPRE) || (*p && *p != '\r' && *p != '\n' && *p != '\t'))
            goto literal_cat;
          if (*p != '\n')
            Strcat_char(buf, ' ');
          break;
      case R_ST_AMP:
      literal_cat:
          Strcat_char(buf, *p);
          break;
      case R_ST_IRRTAG:
          if ((flag & RT_PRE) && prev_status == R_ST_CMNT1) {
            Strcat_char(buf, '!');
            Strcat_char(buf, *p);
            *status = R_ST_TAG;
            break;
          }
      case R_ST_CMNT:
          if (!(flag & RT_APPEND))
            Strclear(buf);
          break;
      case R_ST_CMNT1:
      case R_ST_CMNT2:
      case R_ST_NCMNT1:
      case R_ST_NCMNT2:
      case R_ST_NCMNT3:
          /* do nothing */
          break;
      }
    }
  proc_end:
    *instr = p;
    return 1;
}

Str
correct_irrtag(int status)
{
    char c;
    Str tmp = Strnew();

    while (status != R_ST_NORMAL) {
      switch (status) {
      case R_ST_CMNT:         /* required "-->" */
      case R_ST_NCMNT1: /* required "->" */
          c = '-';
          break;
      case R_ST_NCMNT2:
      case R_ST_NCMNT3:
      case R_ST_IRRTAG:
      case R_ST_CMNT1:
      case R_ST_CMNT2:
      case R_ST_TAG:
      case R_ST_TAG0:
      case R_ST_EQL:          /* required ">" */
          c = '>';
          break;
      case R_ST_QUOTE:
          c = '\'';
          break;
      case R_ST_DQUOTE:
          c = '"';
          break;
      case R_ST_AMP:
          c = ';';
          break;
      default:
          return tmp;
      }
      next_status(c, &status);
      Strcat_char(tmp, c);
    }
    return tmp;
}

static int
dir_under(const char *x, const char *y)
{
    size_t len;
    if (strcmp(x, y) == 0)
      return 1;
    len = strlen(x);
    return len > 0 && x[len - 1] == '/'
      && strlen(y) >= len
      && y[len - 1] == '/' && strncasecmp(x, y, len) == 0;
}

static void
add_auth_pass_entry(const struct auth_pass *ent, int netrc)
{
    if ((ent->host || netrc)  /* netrc accept default (host == NULL) */
      &&(ent->is_proxy || ent->file || ent->realm || netrc)
      && ent->uname && ent->pwd) {
      struct auth_pass *newent = New(struct auth_pass);
      memcpy(newent, ent, sizeof(struct auth_pass));
      if (passwords == NULL)
          passwords = newent;
      else if (passwords->next == NULL)
          passwords->next = newent;
      else {
          struct auth_pass *ep = passwords;
          for (; ep->next; ep = ep->next) ;
          ep->next = newent;
      }
    }
    /* ignore invalid entries */
}

static struct auth_pass *
find_auth_pass_entry(char *host, int port, char *file, char *realm,
                 int is_proxy)
{
    struct auth_pass *ent;
    for (ent = passwords; ent != NULL; ent = ent->next) {
      if (ent->is_proxy == is_proxy
          && (!ent->host || !Strcmp_charp(ent->host, host))
          && (!ent->port || ent->port == port)
          && (!ent->file || !file || dir_under(ent->file->ptr, file))
          && (!ent->realm || !realm || !Strcmp_charp(ent->realm, realm))
          )
          return ent;
    }
    return NULL;
}

int
find_auth_user_passwd(ParsedURL *pu, char *realm,
                  Str *uname, Str *pwd, int is_proxy)
{
    struct auth_pass *ent;

    if (pu->user && pu->pass) {
      *uname = Strnew_charp(pu->user);
      *pwd = Strnew_charp(pu->pass);
      return 1;
    }
    ent = find_auth_pass_entry(pu->host, pu->port, pu->file, realm, is_proxy);
    if (ent) {
      *uname = ent->uname;
      *pwd = ent->pwd;
      return 1;
    }
    return 0;
}

/* passwd */
/*
 * machine <host>
 * host <host>
 * port <port>
 * proxy
 * path <file>
 * realm <realm>
 * login <login>
 * passwd <passwd>
 * password <passwd>
 */

static Str
next_token(Str arg)
{
    Str narg = NULL;
    char *p, *q;
    if (arg == NULL || arg->length == 0)
      return NULL;
    p = arg->ptr;
    q = p;
    SKIP_NON_BLANKS(q);
    if (*q != '\0') {
      *q++ = '\0';
      SKIP_BLANKS(q);
      if (*q != '\0')
          narg = Strnew_charp(q);
    }
    return narg;
}

static void
parsePasswd(FILE * fp, int netrc)
{
    struct auth_pass ent;
    Str line = NULL;

    bzero(&ent, sizeof(struct auth_pass));
    while (1) {
      Str arg = NULL;
      char *p;

      if (line == NULL || line->length == 0)
          line = Strfgets(fp);
      if (line->length == 0)
          break;
      Strchop(line);
      Strremovefirstspaces(line);
      p = line->ptr;
      if (*p == '#' || *p == '\0') {
          line = NULL;
          continue;           /* comment or empty line */
      }
      arg = next_token(line);

      if (!strcmp(p, "machine") || !strcmp(p, "host")
          || (netrc && !strcmp(p, "default"))) {
          add_auth_pass_entry(&ent, netrc);
          bzero(&ent, sizeof(struct auth_pass));
          if (netrc)
            ent.port = 21;    /* XXX: getservbyname("ftp"); ? */
          if (strcmp(p, "default") != 0) {
            line = next_token(arg);
            ent.host = arg;
          }
          else {
            line = arg;
          }
      }
      else if (!netrc && !strcmp(p, "port") && arg) {
          line = next_token(arg);
          ent.port = atoi(arg->ptr);
      }
      else if (!netrc && !strcmp(p, "proxy")) {
          ent.is_proxy = 1;
          line = arg;
      }
      else if (!netrc && !strcmp(p, "path")) {
          line = next_token(arg);
          ent.file = arg;
      }
      else if (!netrc && !strcmp(p, "realm")) {
          /* XXX: rest of line becomes arg for realm */
          line = NULL;
          ent.realm = arg;
      }
      else if (!strcmp(p, "login")) {
          line = next_token(arg);
          ent.uname = arg;
      }
      else if (!strcmp(p, "password") || !strcmp(p, "passwd")) {
          line = next_token(arg);
          ent.pwd = arg;
      }
      else if (netrc && !strcmp(p, "machdef")) {
          while ((line = Strfgets(fp))->length != 0) {
            if (*line->ptr == '\n')
                break;
          }
          line = NULL;
      }
      else if (netrc && !strcmp(p, "account")) {
          /* ignore */
          line = next_token(arg);
      }
      else {
          /* ignore rest of line */
          line = NULL;
      }
    }
    add_auth_pass_entry(&ent, netrc);
}

#define FILE_IS_READABLE_MSG "SECURITY NOTE: file %s must not be accessible by others"

FILE *
openSecretFile(char *fname)
{
    struct stat st;
    if (fname == NULL)
      return NULL;
    if (stat(expandName(fname), &st) < 0)
      return NULL;

    /* check permissions, if group or others readable or writable,
     * refuse it, because it's insecure.
     *
     * XXX: disable_secret_security_check will introduce some
     *    security issues, but on some platform such as Windows
     *    it's not possible (or feasible) to disable group|other
     *    readable and writable.
     *   [w3m-dev 03368][w3m-dev 03369][w3m-dev 03370]
     */
    if (disable_secret_security_check)
      /* do nothing */ ;
    else if ((st.st_mode & (S_IRWXG | S_IRWXO)) != 0) {
      if (fmInitialized) {
          message(Sprintf(FILE_IS_READABLE_MSG, fname)->ptr);
          refresh();
      }
      else {
          fputs(Sprintf(FILE_IS_READABLE_MSG, fname)->ptr, stderr);
          fputc('\n', stderr);
      }
      sleep(2);
      return NULL;
    }

    return fopen(expandName(fname), "r");
}

void
loadPasswd(void)
{
    FILE *fp;

    passwords = NULL;
    fp = openSecretFile(passwd_file);
    if (fp != NULL) {
      parsePasswd(fp, 0);
      fclose(fp);
    }

    /* for FTP */
    fp = openSecretFile("~/.netrc");
    if (fp != NULL) {
      parsePasswd(fp, 1);
      fclose(fp);
    }
    return;
}

/* authentication */
struct auth_cookie *
find_auth(char *host, int port, char *file, char *realm)
{
    struct auth_cookie *p;

    for (p = Auth_cookie; p != NULL; p = p->next) {
      if (!Strcasecmp_charp(p->host, host) &&
          p->port == port &&
          ((realm && !Strcasecmp_charp(p->realm, realm)) ||
           (p->file && file && dir_under(p->file->ptr, file))))
          return p;
    }
    return NULL;
}

Str
find_auth_cookie(char *host, int port, char *file, char *realm)
{
    struct auth_cookie *p = find_auth(host, port, file, realm);
    if (p)
      return p->cookie;
    return NULL;
}

#ifdef AUTH_DEBUG
static void
dump_auth_cookie(void)
{
    if (w3m_debug) {
      FILE *ff = fopen("zzzauth", "a");
      struct auth_cookie *p;

      for (p = Auth_cookie; p != NULL; p = p->next) {
          Str tmp = Sprintf("%s, %d, %s, %s\n", p->host->ptr, p->port,
                        p->file ? (const char *)p->file->ptr : "NULL",
                        p->realm ? (const char *)p->realm->ptr : "NULL");
          fwrite(tmp->ptr, sizeof(char), tmp->length, ff);
      }
      fputc('\n', ff);
      fclose(ff);
    }
}
#endif

void
add_auth_cookie(char *host, int port, char *file, char *realm, Str cookie)
{
    struct auth_cookie *p;

    p = find_auth(host, port, file, realm);
    if (p && (!p->file || !Strcasecmp_charp(p->file, file))) {
      if (realm && p->realm == NULL)
          p->realm = Strnew_charp(realm);
      p->cookie = cookie;
#ifdef AUTH_DEBUG
      dump_auth_cookie();
#endif
      return;
    }
    p = New(struct auth_cookie);
    p->host = Strnew_charp(host);
    p->port = port;
    p->file = file ? Strnew_charp(file) : NULL;
    p->realm = Strnew_charp(realm);
    p->cookie = cookie;
    p->next = Auth_cookie;
    Auth_cookie = p;
#ifdef AUTH_DEBUG
    dump_auth_cookie();
#endif
}

/* get last modified time */
char *
last_modified(Buffer * buf)
{
    if (buf->document_header_table) {
      char *hd;

      return (btri_fast_ci_search_str("Last-modified", buf->document_header_table, (void **)&hd) != bt_failure ?
            hd : "unknown");
    }
    else if (buf->currentURL.scheme == SCM_LOCAL) {
      struct stat st;

      if (stat(buf->currentURL.file, &st) < 0)
          return "unknown";
      return ctime(&st.st_mtime);
    }
    return "unknown";
}

static char roman_num1[] = {
    'i', 'x', 'c', 'm', '*',
};
static char roman_num5[] = {
    'v', 'l', 'd', '*',
};

static Str
romanNum2(int l, int n)
{
    Str s = Strnew();

    switch (n) {
    case 1:
    case 2:
    case 3:
      for (; n > 0; n--)
          Strcat_char(s, roman_num1[l]);
      break;
    case 4:
      Strcat_char(s, roman_num1[l]);
      Strcat_char(s, roman_num5[l]);
      break;
    case 5:
    case 6:
    case 7:
    case 8:
      Strcat_char(s, roman_num5[l]);
      for (n -= 5; n > 0; n--)
          Strcat_char(s, roman_num1[l]);
      break;
    case 9:
      Strcat_char(s, roman_num1[l]);
      Strcat_char(s, roman_num1[l + 1]);
      break;
    }
    return s;
}

Str
romanNumeral(int n)
{
    Str r = Strnew();

    if (n <= 0)
      return r;
    if (n >= 4000) {
      Strcat_charp(r, "**");
      return r;
    }
    Strcat(r, romanNum2(3, n / 1000));
    Strcat(r, romanNum2(2, (n % 1000) / 100));
    Strcat(r, romanNum2(1, (n % 100) / 10));
    Strcat(r, romanNum2(0, n % 10));

    return r;
}

Str
romanAlphabet(int n)
{
    Str r = Strnew();
    int l;
    char buf[14];

    if (n <= 0)
      return r;

    l = 0;
    while (n) {
      buf[l++] = 'a' + (n - 1) % 26;
      n = (n - 1) / 26;
    }
    l--;
    for (; l >= 0; l--)
      Strcat_char(r, buf[l]);

    return r;
}

void
mySystem(char *command, int background)
{
#ifdef __EMX__ /* jsawa */
    if (background){
      Str cmd;

      cmd = Sprintf("%s%s%s",
                  (background & MYSYSTEM_BACKGROUND) ? Sprintf("start /b %s", command)->ptr : command,
                  ((background & MYSYSTEM_NULLSTDIN) ? " <nul" :
                   (background & MYSYSTEM_TTYSTDIN) ? " <con" :
                   ""),
                  ((background & MYSYSTEM_NULLSTDOUT) ? " >nul" :
                   (background & MYSYSTEM_TTYSTDOUT) ? " >con" :
                   ""));
      system(cmd->ptr);
    } else
      system(command);
#else
    if (background) {
      Str cmd;

      cmd = Sprintf("(%s)%s%s%s",
                  command,
                  ((background & MYSYSTEM_NULLSTDIN) ? " </dev/null" :
                   (background & MYSYSTEM_TTYSTDIN) ? " </dev/tty" :
                   ""),
                  ((background & MYSYSTEM_NULLSTDOUT) ? " >/dev/null" :
                   (background & MYSYSTEM_TTYSTDOUT) ? " >/dev/tty 2>&1" :
                   ""),
                  (background & MYSYSTEM_BACKGROUND) ? " &" : "");
      system(cmd->ptr);
    }
    else
      system(command);
#endif
}

Str
myExtCommand(char *cmd, char *arg, int redirect)
{
    Str tmp = NULL;
    char *p;
    int set_arg = FALSE;

    for (p = cmd; *p; p++) {
      if (*p == '%' && *(p + 1) == 's' && !set_arg) {
          if (tmp == NULL)
            tmp = Strnew_charp_n(cmd, (int)(p - cmd));
          Strcat_charp(tmp, arg);
          set_arg = TRUE;
          p++;
      }
      else {
          if (tmp)
            Strcat_char(tmp, *p);
      }
    }
    if (!set_arg)
      tmp = Strnew_m_charp(cmd, (redirect ? " < " : " "), arg, NULL);
    return tmp;
}

char *
expandName(char *name)
{
    Str userName = NULL;
    char *p;
    struct passwd *passent, *getpwnam(const char *);
    Str extpath = Strnew();

    p = name;
    if (*p == '/' && *(p + 1) == '~' && IS_ALPHA(*(p + 2))) {
      if (personal_document_root != NULL) {
          userName = Strnew();
          p += 2;
          while (IS_ALNUM(*p) || *p == '_' || *p == '-')
            Strcat_char(userName, *(p++));
          passent = getpwnam(userName->ptr);
          if (passent == NULL) {
            p = name;
            goto rest;
          }
          Strcat_charp(extpath, passent->pw_dir);
          Strcat_char(extpath, '/');
          Strcat_charp(extpath, personal_document_root);
          if (Strcmp_charp(extpath, "/") == 0 && *p == '/')
            p++;
      }
    }
    else
      p = expandPath(p);
  rest:
    Strcat_charp(extpath, p);
    return extpath->ptr;
}

char *
file_to_url(char *file)
{
   Str tmp;
#ifdef SUPPORT_DOS_DRIVE_PREFIX
   char *drive = NULL;
#endif
#ifdef SUPPORT_NETBIOS_SHARE
   char *host = NULL;
#endif

   file = expandName(file);
#ifdef SUPPORT_NETBIOS_SHARE
   if (file[0] == '/' && file[1] == '/') {
      char *p;
      file += 2;
      if (*file) {
          p = strchr(file, '/');
          if (p != NULL && p != file) {
       host = allocStr(file, (p - file));
       file = p;
          }
      }
   }
#endif
#ifdef SUPPORT_DOS_DRIVE_PREFIX
   if (IS_ALPHA(file[0]) && file[1] == ':') {
      drive = allocStr(file, 2);
      file += 2;
   }
   else
#endif
   if (file[0] != '/') {
      tmp = Strnew_charp(CurrentDir);
      if (Strlastchar(tmp) != '/')
          Strcat_char(tmp, '/');
      Strcat_charp(tmp, file);
      file = tmp->ptr;
   }
   tmp = Strnew_charp("file://");
#ifdef SUPPORT_NETBIOS_SHARE
   if (host)
      Strcat_charp(tmp, host);
#endif
#ifdef SUPPORT_DOS_DRIVE_PREFIX
   if (drive)
      Strcat_charp(tmp, drive);
#endif
   Strcat_charp(tmp, file_quote(cleanupName(file)));
   return tmp->ptr;
}

static char *tmpf_base[MAX_TMPF_TYPE] = {
    "tmp", "src", "frame", "cache" 
};
static unsigned int tmpf_seq[MAX_TMPF_TYPE];
    
Str
tmpfname(int type, char *ext)
{
    Str tmpf;
    tmpf = Sprintf("%s/w3m%s%d-%d%s",
               tmp_dir,
               tmpf_base[type],
               (int)getpid(), tmpf_seq[type]++, (ext)? ext : "");
    return tmpf;
}

#ifdef USE_COOKIE
static char *monthtbl[] = {
    "Jan", "Feb", "Mar", "Apr", "May", "Jun",
    "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"
};

static int
get_day(char **s)
{
    Str tmp = Strnew();
    int day;
    char *ss = *s;

    if (!**s)
      return -1;

    while (**s && IS_DIGIT(**s))
      Strcat_char(tmp, *((*s)++));

    day = atoi(tmp->ptr);

    if (day < 1 || day > 31) {
      *s = ss;
      return -1;
    }
    return day;
}

static int
get_month(char **s)
{
    Str tmp = Strnew();
    int mon;
    char *ss = *s;

    if (!**s)
      return -1;

    while (**s && IS_DIGIT(**s))
      Strcat_char(tmp, *((*s)++));
    if (tmp->length > 0) {
      mon = atoi(tmp->ptr);
    }
    else {
      while (**s && IS_ALPHA(**s))
          Strcat_char(tmp, *((*s)++));
      for (mon = 1; mon <= 12; mon++) {
          if (strncmp(tmp->ptr, monthtbl[mon - 1], 3) == 0)
            break;
      }
    }
    if (mon < 1 || mon > 12) {
      *s = ss;
      return -1;
    }
    return mon;
}

static int
get_year(char **s)
{
    Str tmp = Strnew();
    int year;
    char *ss = *s;

    if (!**s)
      return -1;

    while (**s && IS_DIGIT(**s))
      Strcat_char(tmp, *((*s)++));
    if (tmp->length != 2 && tmp->length != 4) {
      *s = ss;
      return -1;
    }

    year = atoi(tmp->ptr);
    if (tmp->length == 2) {
      if (year >= 70)
          year += 1900;
      else
          year += 2000;
    }
    return year;
}

static int
get_time(char **s, int *hour, int *min, int *sec)
{
    Str tmp = Strnew();
    char *ss = *s;

    if (!**s)
      return -1;

    while (**s && IS_DIGIT(**s))
      Strcat_char(tmp, *((*s)++));
    if (**s != ':') {
      *s = ss;
      return -1;
    }
    *hour = atoi(tmp->ptr);

    (*s)++;
    Strclear(tmp);
    while (**s && IS_DIGIT(**s))
      Strcat_char(tmp, *((*s)++));
    if (**s != ':') {
      *s = ss;
      return -1;
    }
    *min = atoi(tmp->ptr);

    (*s)++;
    Strclear(tmp);
    while (**s && IS_DIGIT(**s))
      Strcat_char(tmp, *((*s)++));
    *sec = atoi(tmp->ptr);

    if (*hour < 0 || *hour >= 24 ||
      *min < 0 || *min >= 60 || *sec < 0 || *sec >= 60) {
      *s = ss;
      return -1;
    }
    return 0;
}

/* RFC 1123 or RFC 850 or ANSI C asctime() format string -> time_t */
time_t
mymktime(char *timestr)
{
    char *s;
    int day, mon, year, hour, min, sec;

    if (!(timestr && *timestr))
      return -1;
    s = timestr;

#ifdef DEBUG
    fprintf(stderr, "mktime: %s\n", timestr);
#endif                        /* DEBUG */

    while (*s && IS_ALPHA(*s))
      s++;
    while (*s && !IS_ALNUM(*s))
      s++;

    if (IS_DIGIT(*s)) {
      /* RFC 1123 or RFC 850 format */
      if ((day = get_day(&s)) == -1)
          return -1;

      while (*s && !IS_ALNUM(*s))
          s++;
      if ((mon = get_month(&s)) == -1)
          return -1;

      while (*s && !IS_DIGIT(*s))
          s++;
      if ((year = get_year(&s)) == -1)
          return -1;

      while (*s && !IS_DIGIT(*s))
          s++;
      if (!*s) {
          hour = 0;
          min = 0;
          sec = 0;
      }
      else if (get_time(&s, &hour, &min, &sec) == -1) {
          return -1;
      }
    }
    else {
      /* ANSI C asctime() format. */
      while (*s && !IS_ALNUM(*s))
          s++;
      if ((mon = get_month(&s)) == -1)
          return -1;

      while (*s && !IS_DIGIT(*s))
          s++;
      if ((day = get_day(&s)) == -1)
          return -1;

      while (*s && !IS_DIGIT(*s))
          s++;
      if (get_time(&s, &hour, &min, &sec) == -1)
          return -1;

      while (*s && !IS_DIGIT(*s))
          s++;
      if ((year = get_year(&s)) == -1)
          return -1;
    }
#ifdef DEBUG
    fprintf(stderr, "year=%d month=%d day=%d hour:min:sec=%d:%d:%d\n",
          year, mon, day, hour, min, sec);
#endif                        /* DEBUG */

    mon -= 3;
    if (mon < 0) {
      mon += 12;
      year--;
    }
    day += (year - 1968) * 1461 / 4;
    day += ((((mon * 153) + 2) / 5) - 672);
    return (time_t) ((day * 60 * 60 * 24) +
                 (hour * 60 * 60) + (min * 60) + sec);
}

#ifdef INET6
#include <sys/socket.h>
#endif                        /* INET6 */
#include <netdb.h>
char *
FQDN(char *host)
{
    char *p;
#ifndef INET6
    struct hostent *entry;
#else                   /* INET6 */
    int *af;
#endif                        /* INET6 */

    if (host == NULL)
      return NULL;

    if (strcasecmp(host, "localhost") == 0)
      return host;

    for (p = host; *p && *p != '.'; p++)
      ;

    if (*p == '.')
      return host;

#ifndef INET6
    if (!(entry = gethostbyname(host)))
      return NULL;

    return allocStr(entry->h_name, -1);
#else                   /* INET6 */
    for (af = ai_family_order_table[DNS_order];; af++) {
      int error;
      struct addrinfo hints;
      struct addrinfo *res, *res0;
      char *namebuf;

      memset(&hints, 0, sizeof(hints));
      hints.ai_family = *af;
      hints.ai_socktype = SOCK_STREAM;
      error = getaddrinfo(host, NULL, &hints, &res0);
      if (error) {
          if (*af == PF_UNSPEC) {
            /* all done */
            break;
          }
          /* try next address family */
          continue;
      }
      for (res = res0; res != NULL; res = res->ai_next) {
          if (res->ai_canonname) {
            /* found */
            namebuf = strdup(res->ai_canonname);
            freeaddrinfo(res0);
            return namebuf;
          }
      }
      freeaddrinfo(res0);
      if (*af == PF_UNSPEC) {
          break;
      }
    }
    /* all failed */
    return NULL;
#endif                        /* INET6 */
}

#endif                        /* USE_COOKIE */

/* Local Variables:    */
/* c-basic-offset: 4   */
/* tab-width: 8        */
/* End:                */

Generated by  Doxygen 1.6.0   Back to index