1 /* ftruncate emulations that work on some System V's.
2    This file is in the public domain.  */
3 
4 /* Copyright (C) 2012-2014 Free Software Foundation, Inc.
5 
6    This file is part of gold.
7 
8    This program is free software; you can redistribute it and/or modify
9    it under the terms of the GNU General Public License as published by
10    the Free Software Foundation; either version 3 of the License, or
11    (at your option) any later version.
12 
13    This program is distributed in the hope that it will be useful,
14    but WITHOUT ANY WARRANTY; without even the implied warranty of
15    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16    GNU General Public License for more details.
17 
18    You should have received a copy of the GNU General Public License
19    along with this program; if not, write to the Free Software
20    Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
21    MA 02110-1301, USA.  */
22 
23 #include <config.h>
24 
25 /* Specification.  */
26 #include <unistd.h>
27 
28 #include <sys/types.h>
29 #include <fcntl.h>
30 
31 extern int ftruncate (int, off_t);
32 
33 #ifdef F_CHSIZE
34 
35 int
ftruncate(int fd,off_t length)36 ftruncate (int fd, off_t length)
37 {
38   return fcntl (fd, F_CHSIZE, length);
39 }
40 
41 #else /* not F_CHSIZE */
42 # ifdef F_FREESP
43 
44 /* By William Kucharski <kucharsk@netcom.com>.  */
45 
46 #  include <sys/stat.h>
47 #  include <errno.h>
48 
49 int
ftruncate(int fd,off_t length)50 ftruncate (int fd, off_t length)
51 {
52   struct flock fl;
53   struct stat filebuf;
54 
55   if (fstat (fd, &filebuf) < 0)
56     return -1;
57 
58   if (filebuf.st_size < length)
59     {
60       /* Extend file length. */
61       if (lseek (fd, (length - 1), SEEK_SET) < 0)
62 	return -1;
63 
64       /* Write a "0" byte. */
65       if (write (fd, "", 1) != 1)
66 	return -1;
67     }
68   else
69     {
70 
71       /* Truncate length. */
72 
73       fl.l_whence = 0;
74       fl.l_len = 0;
75       fl.l_start = length;
76       fl.l_type = F_WRLCK;	/* write lock on file space */
77 
78       /* This relies on the *undocumented* F_FREESP argument to fcntl,
79 	 which truncates the file so that it ends at the position
80 	 indicated by fl.l_start.  Will minor miracles never cease?  */
81 
82       if (fcntl (fd, F_FREESP, &fl) < 0)
83 	return -1;
84     }
85 
86   return 0;
87 }
88 
89 # else /* not F_CHSIZE nor F_FREESP */
90 #  if HAVE_CHSIZE                      /* native Windows, e.g. mingw */
91 
92 int
ftruncate(int fd,off_t length)93 ftruncate (int fd, off_t length)
94 {
95   return chsize (fd, length);
96 }
97 
98 #  else /* not F_CHSIZE nor F_FREESP nor HAVE_CHSIZE */
99 
100 #   include <errno.h>
101 
102 int
ftruncate(int fd,off_t length)103 ftruncate (int fd, off_t length)
104 {
105   errno = EIO;
106   return -1;
107 }
108 
109 #  endif /* not HAVE_CHSIZE */
110 # endif /* not F_FREESP */
111 #endif /* not F_CHSIZE */
112