File: readln.c

package info (click to toggle)
sn 0.3.4a-2
  • links: PTS
  • area: main
  • in suites: woody
  • size: 784 kB
  • ctags: 826
  • sloc: ansic: 9,023; sh: 339; makefile: 208
file content (119 lines) | stat: -rw-r--r-- 2,606 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
/*
 * This file is part of the sn package.
 * Distribution of sn is covered by the GNU GPL. See file COPYING.
 * Copyright  1998-2000 Harold Tay.
 * Copyright  2000- Patrik Rdman.
 */

/*
 * Get a single line from an fd.
 */

#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/time.h>
#include <time.h>
#include <errno.h>
#include "readln.h"

static const char rcsid[] = "$Id$";

int readln_ready (int fd, int tmo, struct readln *rp)
{
   rp->buf = rp->bf;
   rp->size = sizeof (rp->bf);
   rp->fd = fd;
   rp->eaten = rp->used = 0;
   rp->tmo = tmo;
   return (0);
}

void readln_done (struct readln *rp)
{
   if (rp && rp->buf != rp->bf)
      free(rp->buf);
}

int readln (register struct readln *rp, char **line, int ch)
{
   char *endp;
   int len;

   if (rp->eaten == rp->used)
      rp->eaten = rp->used = 0;
   else if (rp->eaten && rp->eaten + 64 >= rp->used)
   {
      register char *from;
      register char *to;
      int n;

      from = rp->buf + rp->eaten;
      to = rp->buf;
      for (n = rp->used - rp->eaten; n; n--)
         *to++ = *from++;
      rp->used -= rp->eaten;
      rp->eaten = 0;
   }

   while (1)
   {
      char *lim;
      int count;

      lim = rp->buf + rp->used;
      for (endp = rp->buf + rp->eaten; endp < lim; endp++)
         if (ch == *endp)
            goto done;
      if (rp->size - rp->used < 16)
      {
         char *tmp;

         if (rp->buf == rp->bf)
            tmp = malloc(rp->size = 504);
         else
            tmp = malloc(rp->size *= 2);
         if (!tmp)
            return (-1);
         memcpy(tmp, rp->buf + rp->eaten, rp->used - rp->eaten);
         rp->used -= rp->eaten;
         rp->eaten = 0;
         if (rp->buf != rp->bf)
            free(rp->buf);
         rp->buf = tmp;
      }
      if (rp->tmo > 0)
      {
         struct timeval tv;
         fd_set set;

         FD_ZERO(&set);
         for (;;)
         {
            FD_SET(rp->fd, &set);
            tv.tv_usec = 0;
            tv.tv_sec = rp->tmo;
            switch (select(rp->fd + 1, &set, 0, 0, &tv))
            {
               case 0:
                  return (-1);
               case -1:
                  if (EINTR == errno)
                     continue;
                  return (-1);
            }
            break;
         }
      }
      count = read(rp->fd, rp->buf + rp->used, rp->size - rp->used);
      if (count <= 0)
         return (count);
      rp->used += count;
   }

done:
   *line = rp->buf + rp->eaten;
   len = 1 + endp - (rp->buf + rp->eaten);
   rp->eaten = 1 + endp - rp->buf;
   return (len);
}