view src/lib-mail/message-send.c @ 420:0de1093e8054 HEAD

check return value of o_buffer_send() correctly
author Timo Sirainen <tss@iki.fi>
date Mon, 14 Oct 2002 20:48:03 +0300
parents 1f0e7229ee58
children 20d5136a0c6a
line wrap: on
line source

/* Copyright (C) 2002 Timo Sirainen */

#include "lib.h"
#include "ibuffer.h"
#include "obuffer.h"
#include "message-send.h"
#include "message-size.h"

int message_send(OBuffer *outbuf, IBuffer *inbuf, MessageSize *msg_size,
		 uoff_t virtual_skip, uoff_t max_virtual_size)
{
	const unsigned char *msg;
	uoff_t old_limit;
	size_t i, size;
	int cr_skipped, add_cr, ret;

	if (msg_size->physical_size == 0 ||
	    virtual_skip >= msg_size->virtual_size)
		return TRUE;

	if (max_virtual_size > msg_size->virtual_size - virtual_skip)
		max_virtual_size = msg_size->virtual_size - virtual_skip;

	if (msg_size->physical_size == msg_size->virtual_size) {
		/* no need to kludge with CRs, we can use sendfile() */
		i_buffer_skip(inbuf, virtual_skip);

		old_limit = inbuf->v_limit;
		i_buffer_set_read_limit(inbuf,
					I_MIN(max_virtual_size, old_limit));
		ret = o_buffer_send_ibuffer(outbuf, inbuf) > 0;
		i_buffer_set_read_limit(inbuf, old_limit);

		return ret;
	}

	message_skip_virtual(inbuf, virtual_skip, NULL, &cr_skipped);

	/* go through the message data and insert CRs where needed.  */
	while (i_buffer_read_data(inbuf, &msg, &size, 0) > 0) {
		add_cr = FALSE;
		for (i = 0; i < size; i++) {
			if (msg[i] == '\n') {
				if ((i == 0 && !cr_skipped) ||
				    (i > 0 && msg[i-1] != '\r')) {
					/* missing CR */
					if (max_virtual_size > 0)
						max_virtual_size--;
					add_cr = TRUE;
					break;
				}

			}

			if (max_virtual_size > 0) {
				if (--max_virtual_size == 0) {
					i++;
					break;
				}
			}
		}

		if (o_buffer_send(outbuf, msg, i) < 0)
			return FALSE;

		if (add_cr) {
			if (o_buffer_send(outbuf, "\r", 1) < 0)
				return FALSE;
			cr_skipped = TRUE;
		} else {
			cr_skipped = i > 0 && msg[i-1] == '\r';
		}

		/* see if we've reached the limit */
		if (max_virtual_size == 0)
			break;

		i_buffer_skip(inbuf, i);
	}

	return TRUE;
}