1 #include <unistd.h>
2 #include <fcntl.h>
3 #include <stdio.h>
4 #include <string.h>
5 #include <stdlib.h>
6 #include <sys/stat.h>
7 
8 #define BUFSIZE (1024*8)
9 static void to_unix(char* buf);
10 static void unix_to_dos(char* buf2, const char* buf);
11 
12 int usage()
13 {
14     fprintf(stderr, "usage: line_endings unix|dos FILES\n"
15             "\n"
16             "Convert FILES to either unix or dos line endings.\n");
17     return 1;
18 }
19 
20 typedef struct Node {
21     struct Node *next;
22     char buf[BUFSIZE*2+3];
23 } Node;
24 
25 int
26 main(int argc, char** argv)
27 {
28     enum { UNIX, DOS } ending;
29     int i;
30 
31     if (argc < 2) {
32         return usage();
33     }
34 
35     if (0 == strcmp("unix", argv[1])) {
36         ending = UNIX;
37     }
38     else if (0 == strcmp("dos", argv[1])) {
39         ending = DOS;
40     }
41     else {
42         return usage();
43     }
44 
45     for (i=2; i<argc; i++) {
46         int fd;
47         int len;
48 
49         // force implied
50         chmod(argv[i], S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP);
51 
52         fd = open(argv[i], O_RDWR);
53         if (fd < 0) {
54             fprintf(stderr, "unable to open file for read/write: %s\n", argv[i]);
55             return 1;
56         }
57 
58         len = lseek(fd, 0, SEEK_END);
59         lseek(fd, 0, SEEK_SET);
60 
61         if (len > 0) {
62             Node* root = malloc(sizeof(Node));
63             Node* node = root;
64             node->buf[0] = 0;
65 
66             while (len > 0) {
67                 node->next = malloc(sizeof(Node));
68                 node = node->next;
69                 node->next = NULL;
70 
71                 char buf[BUFSIZE+2];
72                 ssize_t amt;
73                 ssize_t amt2 = len < BUFSIZE ? len : BUFSIZE;
74                 amt = read(fd, buf, amt2);
75                 if (amt != amt2) {
76                     fprintf(stderr, "unable to read file: %s\n", argv[i]);
77                     return 1;
78                 }
79                 buf[amt2] = '\0';
80                 to_unix(buf);
81                 if (ending == UNIX) {
82                     strcpy(node->buf, buf);
83                 } else {
84                     char buf2[(BUFSIZE*2)+3];
85                     unix_to_dos(buf2, buf);
86                     strcpy(node->buf, buf2);
87                 }
88                 len -= amt2;
89             }
90 
91             (void)ftruncate(fd, 0);
92             lseek(fd, 0, SEEK_SET);
93             while (root) {
94                 ssize_t amt2 = strlen(root->buf);
95                 if (amt2 > 0) {
96                     ssize_t amt = write(fd, root->buf, amt2);
97                     if (amt != amt2) {
98                         fprintf(stderr, "unable to write file: %s\n", argv[i]);
99                         return 1;
100                     }
101                 }
102                 node = root;
103                 root = root->next;
104                 free(node);
105             }
106         }
107         close(fd);
108     }
109     return 0;
110 }
111 
112 void
113 to_unix(char* buf)
114 {
115     char* p = buf;
116     char* q = buf;
117     while (*p) {
118         if (p[0] == '\r' && p[1] == '\n') {
119             // dos
120             *q = '\n';
121             p += 2;
122             q += 1;
123         }
124         else if (p[0] == '\r') {
125             // old mac
126             *q = '\n';
127             p += 1;
128             q += 1;
129         }
130         else {
131             *q = *p;
132             p += 1;
133             q += 1;
134         }
135     }
136     *q = '\0';
137 }
138 
139 void
140 unix_to_dos(char* buf2, const char* buf)
141 {
142     const char* p = buf;
143     char* q = buf2;
144     while (*p) {
145         if (*p == '\n') {
146             q[0] = '\r';
147             q[1] = '\n';
148             q += 2;
149             p += 1;
150         } else {
151             *q = *p;
152             p += 1;
153             q += 1;
154         }
155     }
156     *q = '\0';
157 }
158 
159