diff options
Diffstat (limited to 'libical/src')
-rw-r--r-- | libical/src/libical/icalproperty.c | 111 |
1 files changed, 79 insertions, 32 deletions
diff --git a/libical/src/libical/icalproperty.c b/libical/src/libical/icalproperty.c index 331443e3dd..bd361aecc2 100644 --- a/libical/src/libical/icalproperty.c +++ b/libical/src/libical/icalproperty.c @@ -263,51 +263,91 @@ icalproperty_free (icalproperty* prop) } -/* This splits the property every 75 octects, as described in the spec. +/* This returns where the start of the next line should be. chars_left does + not include the trailing '\0'. */ +static char* +get_next_line_start (char *line_start, int chars_left) +{ + char *pos; + + /* If we have 74 chars or less left, we can output all of them. + we return a pointer to the '\0' at the end of the string. */ + if (chars_left <= 74) { + return line_start + chars_left; + } + + /* Now we jump to the last possible character of the line, and step back + trying to find a ';' ':' or ' '. If we find one, we return the character + after it. If not, we break at 74 chars (the 75th char is the space at + the start of the line). */ + pos = line_start + 73; + while (pos > line_start) { + if (*pos == ';' || *pos == ':' || *pos == ' ') { + return pos + 1; + } + pos--; + } + + return line_start + 74; +} + + +/* This splits the property into lines less than 75 octects long (as specified + in RFC2445). It tries to split after a ';' if it can. It returns a tmp buffer. NOTE: I'm not sure if it matters if we split a line in the middle of a UTF-8 character. It probably won't look nice in a text editor. */ static char* fold_property_line (char *text) { - int len, max_lines, line_length, buf_len; - char *buf, *src, *dest, ch; + size_t buf_size; + char *buf, *buf_ptr, *line_start, *next_line_start, *out_buf; + int len, chars_left, first_line; + char ch; + /* Start with a buffer twice the size of our property line, so we almost + certainly won't overflow it. */ len = strlen (text); - - /* The minimum length we split a line at is 65 characters, so calculate - the maximum number of newlines we will need. */ - max_lines = len / 65; - - /* Calculate the maximum size for the buffer we need, if we add a newline - and a space character for each line, and a '\0' at the end. */ - buf_len = len + (max_lines * 2); - buf = icalmemory_tmp_buffer (buf_len + 1); - - src = text; - dest = buf; - line_length = 0; - while (ch = *src) { - /* If the line is 65 characters or over, try to split at next space. - If it is 75 characters or over, split now. */ - if ((line_length >= 65 && ch == ' ') || (line_length >= 75)) { - *dest++ = '\n'; - *dest++ = ' '; - line_length = 0; + buf_size = len * 2; + buf = icalmemory_new_buffer (buf_size); + buf_ptr = buf; + + /* Step through the text, finding each line to add to the output. */ + line_start = text; + chars_left = len; + first_line = 1; + for (;;) { + if (chars_left <= 0) + break; + + /* This returns the first character for the next line. */ + next_line_start = get_next_line_start (line_start, chars_left); + + /* If this isn't the first line, we need to output a newline and space + first. */ + if (!first_line) { + icalmemory_append_string (&buf, &buf_ptr, &buf_size, "\n "); } + first_line = 0; - *dest++ = ch; - line_length++; - - src++; + /* This adds the line to our tmp buffer. We temporarily place a '\0' + in text, so we can copy the line in one go. */ + ch = *next_line_start; + *next_line_start = '\0'; + icalmemory_append_string (&buf, &buf_ptr, &buf_size, line_start); + *next_line_start = ch; - if (dest - buf > buf_len) { - icalerror_warn ("Buffer overflow."); - } + /* Now we move on to the next line. */ + chars_left -= (next_line_start - line_start); + line_start = next_line_start; } - *dest = '\0'; - return buf; + /* Copy it to a temporary buffer, and then free it. */ + out_buf = icalmemory_tmp_buffer (strlen (buf) + 1); + strcpy (out_buf, buf); + icalmemory_free_buffer (buf); + + return out_buf; } @@ -459,6 +499,13 @@ icalproperty_as_ical_string (icalproperty* prop) /* We now use a function to fold the line properly every 75 characters. */ out_buf = fold_property_line (buf); + /* This is useful for testing. It outputs the property before and after + folding, but only if it was changed. */ +#if 0 + if (strcmp (buf, out_buf)) + printf ("Property:\n%sFolded:\n%s", buf, out_buf); +#endif + /*out_buf = icalmemory_tmp_buffer(strlen(buf)+1);*/ /*strcpy(out_buf, buf);*/ |