-
Notifications
You must be signed in to change notification settings - Fork 2
/
ascii.c
99 lines (93 loc) · 2.09 KB
/
ascii.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
/*
* Part of Very Secure FTPd
* Licence: GPL v2
* Author: Chris Evans
* ascii.c
*
* Routines to handle ASCII mode tranfers. Yuk.
*/
#include "ascii.h"
#pragma CHECKED_SCOPE on
struct ascii_to_bin_ret
vsf_ascii_ascii_to_bin(char *p_buf : count(in_len), unsigned int in_len, int prev_cr)
{
/* Task: translate all \r\n into plain \n. A plain \r not followed by \n must
* not be removed.
*/
unsigned int indexx = 0;
unsigned int written = 0;
_Array_ptr<char> p_out : bounds(p_buf + 1, p_buf + in_len) = 0;
if (prev_cr && (!in_len || p_out[0] != '\n'))
{
p_buf[0] = '\r';
p_out = p_buf;
written++;
}
else
{
p_out = p_buf + 1;
}
int last_was_cr = 0;
while (indexx < in_len)
{
char the_char = p_buf[indexx + 1];
if (the_char != '\r')
{
*p_out++ = the_char;
written++;
}
else if (indexx == in_len - 1)
{
last_was_cr = 1;
}
else if (p_buf[indexx + 2] != '\n')
{
*p_out++ = the_char;
written++;
}
indexx++;
}
struct ascii_to_bin_ret ret = {written, last_was_cr, p_out};
return ret;
}
struct bin_to_ascii_ret
vsf_ascii_bin_to_ascii(const char *p_in : count(in_len), char *p_buf : count(2 * in_len), unsigned int in_len, int prev_cr)
{
/* Task: translate all \n not preceeded by \r into \r\n.
* Note that \r\n stays as \r\n. We used to map it to \r\r\n like wu-ftpd
* but have switched to leaving it, like the more popular proftpd.
*/
_Array_ptr<char> p_out : bounds(p_buf, p_buf + ( 2 * in_len)) = p_buf;
struct bin_to_ascii_ret ret = { 0, 0 };
unsigned int indexx = 0;
unsigned int written = 0;
char last_char = 0;
if (prev_cr)
{
last_char = '\r';
ret.last_was_cr = 1;
}
while (indexx < in_len)
{
char the_char = p_in[indexx];
if (the_char == '\n' && last_char != '\r')
{
*p_out++ = '\r';
written++;
}
*p_out++ = the_char;
written++;
indexx++;
last_char = the_char;
if (the_char == '\r')
{
ret.last_was_cr = 1;
}
else
{
ret.last_was_cr = 0;
}
}
ret.stored = written;
return ret;
}