-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgifinto.c
195 lines (171 loc) · 4.91 KB
/
gifinto.c
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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
/*****************************************************************************
gifinto - save GIF on stdin to file if size over set threshold
*****************************************************************************/
// SPDX-License-Identifier: MIT
// SPDX-File-Copyright-Txt: (C) Copyright 1989 Gershon Elber
#include <ctype.h>
#include <fcntl.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#ifdef _WIN32
#include <io.h>
#else
#include <unistd.h>
#endif /* _WIN32 */
#include "getarg.h"
#include "gif_lib.h"
#define PROGRAM_NAME "gifinto"
#define STRLEN 512
#define DEFAULT_MIN_FILE_SIZE 14 /* More than GIF stamp + screen desc. */
#define DEFAULT_OUT_NAME "GifInto.Gif"
#define DEFAULT_TMP_NAME "TempInto.XXXXXX"
static char *VerStr = PROGRAM_NAME VERSION_COOKIE __DATE__ ", " __TIME__ "\n";
static char *CtrlStr = PROGRAM_NAME " v%- s%-MinFileSize!d h%- GifFile!*s";
static int MinFileSize = DEFAULT_MIN_FILE_SIZE;
#ifdef _WIN32
#include <errno.h>
#include <sys/stat.h>
int mkstemp(char *tpl) {
int fd = -1;
char *p;
int e = errno;
errno = 0;
p = _mktemp(tpl);
if (*p && errno == 0) {
errno = e;
fd = _open(p, _O_RDWR | _O_CREAT | _O_EXCL | _O_BINARY,
_S_IREAD | _S_IWRITE);
}
return fd;
}
#endif
/******************************************************************************
This is simply: read until EOF, then close the output, test its length, and
if non zero then rename it.
******************************************************************************/
int main(int argc, char **argv) {
int FD;
int NumFiles;
bool Error, MinSizeFlag = false, HelpFlag = false,
GifNoisyPrint = false;
char **FileName = NULL, FoutTmpName[STRLEN + 1], FullPath[STRLEN + 1],
*p;
FILE *Fin, *Fout;
if ((Error = GAGetArgs(argc, argv, CtrlStr, &GifNoisyPrint,
&MinSizeFlag, &MinFileSize, &HelpFlag, &NumFiles,
&FileName)) != false ||
(NumFiles > 1 && !HelpFlag)) {
if (Error) {
GAPrintErrMsg(Error);
} else if (NumFiles != 1) {
GIF_MESSAGE("Error in command line parsing - one GIF "
"file please.");
}
GAPrintHowTo(CtrlStr);
exit(EXIT_FAILURE);
}
if (HelpFlag) {
(void)fprintf(stderr, VerStr, GIFLIB_MAJOR, GIFLIB_MINOR);
GAPrintHowTo(CtrlStr);
exit(EXIT_SUCCESS);
}
/* Open the stdin in binary mode and increase its buffer size: */
#ifdef _WIN32
_setmode(0, O_BINARY); /* Make sure it is in binary mode. */
#endif
Fin = fdopen(0, "rb"); /* Make it into a stream: */
if (Fin == NULL) {
GIF_EXIT("Failed to open input.");
}
/* Isolate the directory where our destination is, and set tmp file name
*/
/* in the very same directory. This code is isecure because it creates
*/
/* predictable names, but it's not worth the effort and risk to fix. */
if (*FileName == NULL) {
GIF_EXIT("No valid Filename given.");
}
if (strlen(*FileName) > STRLEN - 1) {
GIF_EXIT("Filename too long.");
}
memset(FullPath, '\0', sizeof(FullPath));
strncpy(FullPath, *FileName, STRLEN);
if ((p = strrchr(FullPath, '/')) != NULL ||
(p = strrchr(FullPath, '\\')) != NULL) {
p[1] = 0;
} else if ((p = strrchr(FullPath, ':')) != NULL) {
p[1] = 0;
} else {
FullPath[0] = 0; /* No directory or disk specified. */
}
if (strlen(FullPath) > STRLEN - 1) {
GIF_EXIT("Filename too long.");
}
strncpy(FoutTmpName, FullPath, STRLEN); /* First setup the Path */
/* then add a name for the tempfile */
if ((strlen(FoutTmpName) + strlen(DEFAULT_TMP_NAME)) > STRLEN - 1) {
GIF_EXIT("Filename too long.");
}
strcat(FoutTmpName, DEFAULT_TMP_NAME);
#ifdef _WIN32
char *tmpFN = _mktemp(FoutTmpName);
if (tmpFN) {
FD = open(tmpFN, O_CREAT | O_EXCL | O_WRONLY);
} else {
FD = -1;
}
#else
FD = mkstemp(FoutTmpName); /* returns filedescriptor */
#endif
if (FD == -1) {
GIF_EXIT("Failed to open output.");
}
Fout = fdopen(FD, "wb"); /* returns a stream with FD */
if (Fout == NULL) {
GIF_EXIT("Failed to open output.");
}
while (1) {
int c = getc(Fin);
if (feof(Fin)) {
break;
}
if (putc(c, Fout) == EOF) {
GIF_EXIT("Failed to write output.");
}
}
fclose(Fin);
if (ftell(Fout) >= (long)MinFileSize) {
fclose(Fout);
unlink(*FileName);
if (rename(FoutTmpName, *FileName) != 0) {
char DefaultName[STRLEN + 1];
memset(DefaultName, '\0', sizeof(DefaultName));
if ((strlen(FullPath) + strlen(DEFAULT_OUT_NAME)) >
STRLEN - 1) {
GIF_EXIT("Filename too long.");
}
strncpy(DefaultName, FullPath, STRLEN);
strcat(DefaultName, DEFAULT_OUT_NAME);
if (rename(FoutTmpName, DefaultName) == 0) {
char s[STRLEN];
snprintf(
s, STRLEN,
"Failed to rename out file - left as %s.",
DefaultName);
GIF_MESSAGE(s);
} else {
unlink(FoutTmpName);
GIF_MESSAGE(
"Failed to rename out file - deleted.");
}
}
} else {
fclose(Fout);
unlink(FoutTmpName);
GIF_MESSAGE("File too small - not renamed.");
}
return 0;
}
/* end */