Merge from vendor branch GCC:
[dragonfly.git] / contrib / sendmail-8.13.4 / sendmail / bf.c
1 /*
2  * Copyright (c) 1999-2002, 2004 Sendmail, Inc. and its suppliers.
3  *      All rights reserved.
4  *
5  * By using this file, you agree to the terms and conditions set
6  * forth in the LICENSE file which can be found at the top level of
7  * the sendmail distribution.
8  *
9  * Contributed by Exactis.com, Inc.
10  *
11  */
12
13 /*
14 **  This is in transition. Changed from the original bf_torek.c code
15 **  to use sm_io function calls directly rather than through stdio
16 **  translation layer. Will be made a built-in file type of libsm
17 **  next (once safeopen() linkable from libsm).
18 */
19
20 #include <sm/gen.h>
21 SM_RCSID("@(#)$Id: bf.c,v 8.61 2004/08/03 23:59:02 ca Exp $")
22
23 #include <sys/types.h>
24 #include <sys/stat.h>
25 #include <sys/uio.h>
26 #include <fcntl.h>
27 #include <unistd.h>
28 #include <stdlib.h>
29 #include <string.h>
30 #include <errno.h>
31 #include "sendmail.h"
32 #include "bf.h"
33
34 #include <syslog.h>
35
36 /* bf io functions */
37 static ssize_t  sm_bfread __P((SM_FILE_T *, char *, size_t));
38 static ssize_t  sm_bfwrite __P((SM_FILE_T *, const char *, size_t));
39 static off_t    sm_bfseek __P((SM_FILE_T *, off_t, int));
40 static int      sm_bfclose __P((SM_FILE_T *));
41 static int      sm_bfcommit __P((SM_FILE_T *));
42 static int      sm_bftruncate __P((SM_FILE_T *));
43
44 static int      sm_bfopen __P((SM_FILE_T *, const void *, int, const void *));
45 static int      sm_bfsetinfo __P((SM_FILE_T *, int , void *));
46 static int      sm_bfgetinfo __P((SM_FILE_T *, int , void *));
47
48 /*
49 **  Data structure for storing information about each buffered file
50 **  (Originally in sendmail/bf_torek.h for the curious.)
51 */
52
53 struct bf
54 {
55         bool    bf_committed;   /* Has this buffered file been committed? */
56         bool    bf_ondisk;      /* On disk: committed or buffer overflow */
57         long    bf_flags;
58         int     bf_disk_fd;     /* If on disk, associated file descriptor */
59         char    *bf_buf;        /* Memory buffer */
60         int     bf_bufsize;     /* Length of above buffer */
61         int     bf_buffilled;   /* Bytes of buffer actually filled */
62         char    *bf_filename;   /* Name of buffered file, if ever committed */
63         MODE_T  bf_filemode;    /* Mode of buffered file, if ever committed */
64         off_t   bf_offset;      /* Currect file offset */
65         int     bf_size;        /* Total current size of file */
66 };
67
68 #ifdef BF_STANDALONE
69 # define OPEN(fn, omode, cmode, sff) open(fn, omode, cmode)
70 #else /* BF_STANDALONE */
71 # define OPEN(fn, omode, cmode, sff) safeopen(fn, omode, cmode, sff)
72 #endif /* BF_STANDALONE */
73
74 struct bf_info
75 {
76         char    *bi_filename;
77         MODE_T  bi_fmode;
78         size_t  bi_bsize;
79         long    bi_flags;
80 };
81
82 /*
83 **  SM_BFOPEN -- the "base" open function called by sm_io_open() for the
84 **              internal, file-type-specific info setup.
85 **
86 **      Parameters:
87 **              fp -- file pointer being filled-in for file being open'd
88 **              info -- information about file being opened
89 **              flags -- ignored
90 **              rpool -- ignored (currently)
91 **
92 **      Returns:
93 **              Failure: -1 and sets errno
94 **              Success: 0 (zero)
95 */
96
97 static int
98 sm_bfopen(fp, info, flags, rpool)
99         SM_FILE_T *fp;
100         const void *info;
101         int flags;
102         const void *rpool;
103 {
104         char *filename;
105         MODE_T fmode;
106         size_t bsize;
107         long sflags;
108         struct bf *bfp;
109         int l;
110         struct stat st;
111
112         filename = ((struct bf_info *) info)->bi_filename;
113         fmode = ((struct bf_info *) info)->bi_fmode;
114         bsize = ((struct bf_info *) info)->bi_bsize;
115         sflags = ((struct bf_info *) info)->bi_flags;
116
117         /* Sanity checks */
118         if (*filename == '\0')
119         {
120                 /* Empty filename string */
121                 errno = ENOENT;
122                 return -1;
123         }
124         if (stat(filename, &st) == 0)
125         {
126                 /* File already exists on disk */
127                 errno = EEXIST;
128                 return -1;
129         }
130
131         /* Allocate memory */
132         bfp = (struct bf *) sm_malloc(sizeof(struct bf));
133         if (bfp == NULL)
134         {
135                 errno = ENOMEM;
136                 return -1;
137         }
138
139         /* Assign data buffer */
140         /* A zero bsize is valid, just don't allocate memory */
141         if (bsize > 0)
142         {
143                 bfp->bf_buf = (char *) sm_malloc(bsize);
144                 if (bfp->bf_buf == NULL)
145                 {
146                         bfp->bf_bufsize = 0;
147                         sm_free(bfp);
148                         errno = ENOMEM;
149                         return -1;
150                 }
151         }
152         else
153                 bfp->bf_buf = NULL;
154
155         /* Nearly home free, just set all the parameters now */
156         bfp->bf_committed = false;
157         bfp->bf_ondisk = false;
158         bfp->bf_flags = sflags;
159         bfp->bf_bufsize = bsize;
160         bfp->bf_buffilled = 0;
161         l = strlen(filename) + 1;
162         bfp->bf_filename = (char *) sm_malloc(l);
163         if (bfp->bf_filename == NULL)
164         {
165                 if (bfp->bf_buf != NULL)
166                         sm_free(bfp->bf_buf);
167                 sm_free(bfp);
168                 errno = ENOMEM;
169                 return -1;
170         }
171         (void) sm_strlcpy(bfp->bf_filename, filename, l);
172         bfp->bf_filemode = fmode;
173         bfp->bf_offset = 0;
174         bfp->bf_size = 0;
175         bfp->bf_disk_fd = -1;
176         fp->f_cookie = bfp;
177
178         if (tTd(58, 8))
179                 sm_dprintf("sm_bfopen(%s)\n", filename);
180
181         return 0;
182 }
183
184 /*
185 **  BFOPEN -- create a new buffered file
186 **
187 **      Parameters:
188 **              filename -- the file's name
189 **              fmode -- what mode the file should be created as
190 **              bsize -- amount of buffer space to allocate (may be 0)
191 **              flags -- if running under sendmail, passed directly to safeopen
192 **
193 **      Returns:
194 **              a SM_FILE_T * which may then be used with stdio functions,
195 **              or NULL on failure. SM_FILE_T * is opened for writing
196 **              "SM_IO_WHAT_VECTORS").
197 **
198 **      Side Effects:
199 **              none.
200 **
201 **      Sets errno:
202 **              any value of errno specified by sm_io_setinfo_type()
203 **              any value of errno specified by sm_io_open()
204 **              any value of errno specified by sm_io_setinfo()
205 */
206
207 #ifdef __STDC__
208 /*
209 **  XXX This is a temporary hack since MODE_T on HP-UX 10.x is short.
210 **      If we use K&R here, the compiler will complain about
211 **      Inconsistent parameter list declaration
212 **      due to the change from short to int.
213 */
214
215 SM_FILE_T *
216 bfopen(char *filename, MODE_T fmode, size_t bsize, long flags)
217 #else /* __STDC__ */
218 SM_FILE_T *
219 bfopen(filename, fmode, bsize, flags)
220         char *filename;
221         MODE_T fmode;
222         size_t bsize;
223         long flags;
224 #endif /* __STDC__ */
225 {
226         MODE_T omask;
227         SM_FILE_T SM_IO_SET_TYPE(vector, BF_FILE_TYPE, sm_bfopen, sm_bfclose,
228                 sm_bfread, sm_bfwrite, sm_bfseek, sm_bfgetinfo, sm_bfsetinfo,
229                 SM_TIME_FOREVER);
230         struct bf_info info;
231
232         /*
233         **  Apply current umask to fmode as it may change by the time
234         **  the file is actually created.  fmode becomes the true
235         **  permissions of the file, which OPEN() must obey.
236         */
237
238         omask = umask(0);
239         fmode &= ~omask;
240         (void) umask(omask);
241
242         SM_IO_INIT_TYPE(vector, BF_FILE_TYPE, sm_bfopen, sm_bfclose,
243                 sm_bfread, sm_bfwrite, sm_bfseek, sm_bfgetinfo, sm_bfsetinfo,
244                 SM_TIME_FOREVER);
245         info.bi_filename = filename;
246         info.bi_fmode = fmode;
247         info.bi_bsize = bsize;
248         info.bi_flags = flags;
249
250         return sm_io_open(&vector, SM_TIME_DEFAULT, &info, SM_IO_RDWR, NULL);
251 }
252
253 /*
254 **  SM_BFGETINFO -- returns info about an open file pointer
255 **
256 **      Parameters:
257 **              fp -- file pointer to get info about
258 **              what -- type of info to obtain
259 **              valp -- thing to return the info in
260 */
261
262 static int
263 sm_bfgetinfo(fp, what, valp)
264         SM_FILE_T *fp;
265         int what;
266         void *valp;
267 {
268         struct bf *bfp;
269
270         bfp = (struct bf *) fp->f_cookie;
271         switch (what)
272         {
273           case SM_IO_WHAT_FD:
274                 return bfp->bf_disk_fd;
275           case SM_IO_WHAT_SIZE:
276                 return bfp->bf_size;
277           default:
278                 return -1;
279         }
280 }
281
282 /*
283 **  SM_BFCLOSE -- close a buffered file
284 **
285 **      Parameters:
286 **              fp -- cookie of file to close
287 **
288 **      Returns:
289 **              0 to indicate success
290 **
291 **      Side Effects:
292 **              deletes backing file, sm_frees memory.
293 **
294 **      Sets errno:
295 **              never.
296 */
297
298 static int
299 sm_bfclose(fp)
300         SM_FILE_T *fp;
301 {
302         struct bf *bfp;
303
304         /* Cast cookie back to correct type */
305         bfp = (struct bf *) fp->f_cookie;
306
307         /* Need to clean up the file */
308         if (bfp->bf_ondisk && !bfp->bf_committed)
309                 unlink(bfp->bf_filename);
310         sm_free(bfp->bf_filename);
311
312         if (bfp->bf_disk_fd != -1)
313                 close(bfp->bf_disk_fd);
314
315         /* Need to sm_free the buffer */
316         if (bfp->bf_bufsize > 0)
317                 sm_free(bfp->bf_buf);
318
319         /* Finally, sm_free the structure */
320         sm_free(bfp);
321         return 0;
322 }
323
324 /*
325 **  SM_BFREAD -- read a buffered file
326 **
327 **      Parameters:
328 **              cookie -- cookie of file to read
329 **              buf -- buffer to fill
330 **              nbytes -- how many bytes to read
331 **
332 **      Returns:
333 **              number of bytes read or -1 indicate failure
334 **
335 **      Side Effects:
336 **              none.
337 **
338 */
339
340 static ssize_t
341 sm_bfread(fp, buf, nbytes)
342         SM_FILE_T *fp;
343         char *buf;
344         size_t nbytes;
345 {
346         struct bf *bfp;
347         ssize_t count = 0;      /* Number of bytes put in buf so far */
348         int retval;
349
350         /* Cast cookie back to correct type */
351         bfp = (struct bf *) fp->f_cookie;
352
353         if (bfp->bf_offset < bfp->bf_buffilled)
354         {
355                 /* Need to grab some from buffer */
356                 count = nbytes;
357                 if ((bfp->bf_offset + count) > bfp->bf_buffilled)
358                         count = bfp->bf_buffilled - bfp->bf_offset;
359
360                 memcpy(buf, bfp->bf_buf + bfp->bf_offset, count);
361         }
362
363         if ((bfp->bf_offset + nbytes) > bfp->bf_buffilled)
364         {
365                 /* Need to grab some from file */
366                 if (!bfp->bf_ondisk)
367                 {
368                         /* Oops, the file doesn't exist. EOF. */
369                         if (tTd(58, 8))
370                                 sm_dprintf("sm_bfread(%s): to disk\n",
371                                            bfp->bf_filename);
372                         goto finished;
373                 }
374
375                 /* Catch a read() on an earlier failed write to disk */
376                 if (bfp->bf_disk_fd < 0)
377                 {
378                         errno = EIO;
379                         return -1;
380                 }
381
382                 if (lseek(bfp->bf_disk_fd,
383                           bfp->bf_offset + count, SEEK_SET) < 0)
384                 {
385                         if ((errno == EINVAL) || (errno == ESPIPE))
386                         {
387                                 /*
388                                 **  stdio won't be expecting these
389                                 **  errnos from read()! Change them
390                                 **  into something it can understand.
391                                 */
392
393                                 errno = EIO;
394                         }
395                         return -1;
396                 }
397
398                 while (count < nbytes)
399                 {
400                         retval = read(bfp->bf_disk_fd,
401                                       buf + count,
402                                       nbytes - count);
403                         if (retval < 0)
404                         {
405                                 /* errno is set implicitly by read() */
406                                 return -1;
407                         }
408                         else if (retval == 0)
409                                 goto finished;
410                         else
411                                 count += retval;
412                 }
413         }
414
415 finished:
416         bfp->bf_offset += count;
417         return count;
418 }
419
420 /*
421 **  SM_BFSEEK -- seek to a position in a buffered file
422 **
423 **      Parameters:
424 **              fp     -- fp of file to seek
425 **              offset -- position to seek to
426 **              whence -- how to seek
427 **
428 **      Returns:
429 **              new file offset or -1 indicate failure
430 **
431 **      Side Effects:
432 **              none.
433 **
434 */
435
436 static off_t
437 sm_bfseek(fp, offset, whence)
438         SM_FILE_T *fp;
439         off_t offset;
440         int whence;
441
442 {
443         struct bf *bfp;
444
445         /* Cast cookie back to correct type */
446         bfp = (struct bf *) fp->f_cookie;
447
448         switch (whence)
449         {
450           case SEEK_SET:
451                 bfp->bf_offset = offset;
452                 break;
453
454           case SEEK_CUR:
455                 bfp->bf_offset += offset;
456                 break;
457
458           case SEEK_END:
459                 bfp->bf_offset = bfp->bf_size + offset;
460                 break;
461
462           default:
463                 errno = EINVAL;
464                 return -1;
465         }
466         return bfp->bf_offset;
467 }
468
469 /*
470 **  SM_BFWRITE -- write to a buffered file
471 **
472 **      Parameters:
473 **              fp -- fp of file to write
474 **              buf -- data buffer
475 **              nbytes -- how many bytes to write
476 **
477 **      Returns:
478 **              number of bytes written or -1 indicate failure
479 **
480 **      Side Effects:
481 **              may create backing file if over memory limit for file.
482 **
483 */
484
485 static ssize_t
486 sm_bfwrite(fp, buf, nbytes)
487         SM_FILE_T *fp;
488         const char *buf;
489         size_t nbytes;
490 {
491         struct bf *bfp;
492         ssize_t count = 0;      /* Number of bytes written so far */
493         int retval;
494
495         /* Cast cookie back to correct type */
496         bfp = (struct bf *) fp->f_cookie;
497
498         /* If committed, go straight to disk */
499         if (bfp->bf_committed)
500         {
501                 if (lseek(bfp->bf_disk_fd, bfp->bf_offset, SEEK_SET) < 0)
502                 {
503                         if ((errno == EINVAL) || (errno == ESPIPE))
504                         {
505                                 /*
506                                 **  stdio won't be expecting these
507                                 **  errnos from write()! Change them
508                                 **  into something it can understand.
509                                 */
510
511                                 errno = EIO;
512                         }
513                         return -1;
514                 }
515
516                 count = write(bfp->bf_disk_fd, buf, nbytes);
517                 if (count < 0)
518                 {
519                         /* errno is set implicitly by write() */
520                         return -1;
521                 }
522                 goto finished;
523         }
524
525         if (bfp->bf_offset < bfp->bf_bufsize)
526         {
527                 /* Need to put some in buffer */
528                 count = nbytes;
529                 if ((bfp->bf_offset + count) > bfp->bf_bufsize)
530                         count = bfp->bf_bufsize - bfp->bf_offset;
531
532                 memcpy(bfp->bf_buf + bfp->bf_offset, buf, count);
533                 if ((bfp->bf_offset + count) > bfp->bf_buffilled)
534                         bfp->bf_buffilled = bfp->bf_offset + count;
535         }
536
537         if ((bfp->bf_offset + nbytes) > bfp->bf_bufsize)
538         {
539                 /* Need to put some in file */
540                 if (!bfp->bf_ondisk)
541                 {
542                         MODE_T omask;
543
544                         /* Clear umask as bf_filemode are the true perms */
545                         omask = umask(0);
546                         retval = OPEN(bfp->bf_filename,
547                                       O_RDWR | O_CREAT | O_TRUNC | QF_O_EXTRA,
548                                       bfp->bf_filemode, bfp->bf_flags);
549                         (void) umask(omask);
550
551                         /* Couldn't create file: failure */
552                         if (retval < 0)
553                         {
554                                 /*
555                                 **  stdio may not be expecting these
556                                 **  errnos from write()! Change to
557                                 **  something which it can understand.
558                                 **  Note that ENOSPC and EDQUOT are saved
559                                 **  because they are actually valid for
560                                 **  write().
561                                 */
562
563                                 if (!(errno == ENOSPC
564 #ifdef EDQUOT
565                                       || errno == EDQUOT
566 #endif /* EDQUOT */
567                                      ))
568                                         errno = EIO;
569
570                                 return -1;
571                         }
572                         bfp->bf_disk_fd = retval;
573                         bfp->bf_ondisk = true;
574                 }
575
576                 /* Catch a write() on an earlier failed write to disk */
577                 if (bfp->bf_ondisk && bfp->bf_disk_fd < 0)
578                 {
579                         errno = EIO;
580                         return -1;
581                 }
582
583                 if (lseek(bfp->bf_disk_fd,
584                           bfp->bf_offset + count, SEEK_SET) < 0)
585                 {
586                         if ((errno == EINVAL) || (errno == ESPIPE))
587                         {
588                                 /*
589                                 **  stdio won't be expecting these
590                                 **  errnos from write()! Change them into
591                                 **  something which it can understand.
592                                 */
593
594                                 errno = EIO;
595                         }
596                         return -1;
597                 }
598
599                 while (count < nbytes)
600                 {
601                         retval = write(bfp->bf_disk_fd, buf + count,
602                                        nbytes - count);
603                         if (retval < 0)
604                         {
605                                 /* errno is set implicitly by write() */
606                                 return -1;
607                         }
608                         else
609                                 count += retval;
610                 }
611         }
612
613 finished:
614         bfp->bf_offset += count;
615         if (bfp->bf_offset > bfp->bf_size)
616                 bfp->bf_size = bfp->bf_offset;
617         return count;
618 }
619
620 /*
621 **  BFREWIND -- rewinds the SM_FILE_T *
622 **
623 **      Parameters:
624 **              fp -- SM_FILE_T * to rewind
625 **
626 **      Returns:
627 **              0 on success, -1 on error
628 **
629 **      Side Effects:
630 **              rewinds the SM_FILE_T * and puts it into read mode. Normally
631 **              one would bfopen() a file, write to it, then bfrewind() and
632 **              fread(). If fp is not a buffered file, this is equivalent to
633 **              rewind().
634 **
635 **      Sets errno:
636 **              any value of errno specified by sm_io_rewind()
637 */
638
639 int
640 bfrewind(fp)
641         SM_FILE_T *fp;
642 {
643         (void) sm_io_flush(fp, SM_TIME_DEFAULT);
644         sm_io_clearerr(fp); /* quicker just to do it */
645         return sm_io_seek(fp, SM_TIME_DEFAULT, 0, SM_IO_SEEK_SET);
646 }
647
648 /*
649 **  SM_BFCOMMIT -- "commits" the buffered file
650 **
651 **      Parameters:
652 **              fp -- SM_FILE_T * to commit to disk
653 **
654 **      Returns:
655 **              0 on success, -1 on error
656 **
657 **      Side Effects:
658 **              Forces the given SM_FILE_T * to be written to disk if it is not
659 **              already, and ensures that it will be kept after closing. If
660 **              fp is not a buffered file, this is a no-op.
661 **
662 **      Sets errno:
663 **              any value of errno specified by open()
664 **              any value of errno specified by write()
665 **              any value of errno specified by lseek()
666 */
667
668 static int
669 sm_bfcommit(fp)
670         SM_FILE_T *fp;
671 {
672         struct bf *bfp;
673         int retval;
674         int byteswritten;
675
676         /* Get associated bf structure */
677         bfp = (struct bf *) fp->f_cookie;
678
679         /* If already committed, noop */
680         if (bfp->bf_committed)
681                 return 0;
682
683         /* Do we need to open a file? */
684         if (!bfp->bf_ondisk)
685         {
686                 int save_errno;
687                 MODE_T omask;
688                 struct stat st;
689
690                 if (tTd(58, 8))
691                 {
692                         sm_dprintf("bfcommit(%s): to disk\n", bfp->bf_filename);
693                         if (tTd(58, 32))
694                                 sm_dprintf("bfcommit(): filemode %o flags %ld\n",
695                                            bfp->bf_filemode, bfp->bf_flags);
696                 }
697
698                 if (stat(bfp->bf_filename, &st) == 0)
699                 {
700                         errno = EEXIST;
701                         return -1;
702                 }
703
704                 /* Clear umask as bf_filemode are the true perms */
705                 omask = umask(0);
706                 retval = OPEN(bfp->bf_filename,
707                               O_RDWR | O_CREAT | O_EXCL | QF_O_EXTRA,
708                               bfp->bf_filemode, bfp->bf_flags);
709                 save_errno = errno;
710                 (void) umask(omask);
711
712                 /* Couldn't create file: failure */
713                 if (retval < 0)
714                 {
715                         /* errno is set implicitly by open() */
716                         errno = save_errno;
717                         return -1;
718                 }
719
720                 bfp->bf_disk_fd = retval;
721                 bfp->bf_ondisk = true;
722         }
723
724         /* Write out the contents of our buffer, if we have any */
725         if (bfp->bf_buffilled > 0)
726         {
727                 byteswritten = 0;
728
729                 if (lseek(bfp->bf_disk_fd, 0, SEEK_SET) < 0)
730                 {
731                         /* errno is set implicitly by lseek() */
732                         return -1;
733                 }
734
735                 while (byteswritten < bfp->bf_buffilled)
736                 {
737                         retval = write(bfp->bf_disk_fd,
738                                        bfp->bf_buf + byteswritten,
739                                        bfp->bf_buffilled - byteswritten);
740                         if (retval < 0)
741                         {
742                                 /* errno is set implicitly by write() */
743                                 return -1;
744                         }
745                         else
746                                 byteswritten += retval;
747                 }
748         }
749         bfp->bf_committed = true;
750
751         /* Invalidate buf; all goes to file now */
752         bfp->bf_buffilled = 0;
753         if (bfp->bf_bufsize > 0)
754         {
755                 /* Don't need buffer anymore; free it */
756                 bfp->bf_bufsize = 0;
757                 sm_free(bfp->bf_buf);
758         }
759         return 0;
760 }
761
762 /*
763 **  SM_BFTRUNCATE -- rewinds and truncates the SM_FILE_T *
764 **
765 **      Parameters:
766 **              fp -- SM_FILE_T * to truncate
767 **
768 **      Returns:
769 **              0 on success, -1 on error
770 **
771 **      Side Effects:
772 **              rewinds the SM_FILE_T *, truncates it to zero length, and puts
773 **              it into write mode.
774 **
775 **      Sets errno:
776 **              any value of errno specified by fseek()
777 **              any value of errno specified by ftruncate()
778 */
779
780 static int
781 sm_bftruncate(fp)
782         SM_FILE_T *fp;
783 {
784         struct bf *bfp;
785
786         if (bfrewind(fp) < 0)
787                 return -1;
788
789         /* Get bf structure */
790         bfp = (struct bf *) fp->f_cookie;
791         bfp->bf_buffilled = 0;
792         bfp->bf_size = 0;
793
794         /* Need to zero the buffer */
795         if (bfp->bf_bufsize > 0)
796                 memset(bfp->bf_buf, '\0', bfp->bf_bufsize);
797         if (bfp->bf_ondisk)
798         {
799 #if NOFTRUNCATE
800                 /* XXX: Not much we can do except rewind it */
801                 errno = EINVAL;
802                 return -1;
803 #else /* NOFTRUNCATE */
804                 return ftruncate(bfp->bf_disk_fd, 0);
805 #endif /* NOFTRUNCATE */
806         }
807         return 0;
808 }
809
810 /*
811 **  SM_BFSETINFO -- set/change info for an open file pointer
812 **
813 **      Parameters:
814 **              fp -- file pointer to get info about
815 **              what -- type of info to set/change
816 **              valp -- thing to set/change the info to
817 **
818 */
819
820 static int
821 sm_bfsetinfo(fp, what, valp)
822         SM_FILE_T *fp;
823         int what;
824         void *valp;
825 {
826         struct bf *bfp;
827         int bsize;
828
829         /* Get bf structure */
830         bfp = (struct bf *) fp->f_cookie;
831         switch (what)
832         {
833           case SM_BF_SETBUFSIZE:
834                 bsize = *((int *) valp);
835                 bfp->bf_bufsize = bsize;
836
837                 /* A zero bsize is valid, just don't allocate memory */
838                 if (bsize > 0)
839                 {
840                         bfp->bf_buf = (char *) sm_malloc(bsize);
841                         if (bfp->bf_buf == NULL)
842                         {
843                                 bfp->bf_bufsize = 0;
844                                 errno = ENOMEM;
845                                 return -1;
846                         }
847                 }
848                 else
849                         bfp->bf_buf = NULL;
850                 return 0;
851           case SM_BF_COMMIT:
852                 return sm_bfcommit(fp);
853           case SM_BF_TRUNCATE:
854                 return sm_bftruncate(fp);
855           case SM_BF_TEST:
856                 return 1; /* always */
857           default:
858                 errno = EINVAL;
859                 return -1;
860         }
861 }