/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * Authors: Jeffrey Stedfast * Michael Zucchi * * Copyright 2001 Ximian, Inc. (www.ximian.com) * * This program is free software; you can redistribute it and/or * modify it under the terms of version 2 of the GNU General Public * License as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public * License along with this program; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. * */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include "em-camel-stream.h" #include "mail-mt.h" #define EMCS_BUFFER_SIZE (4096) #define LOG_STREAM 1 #define d(x) enum _write_msg_t { EMCS_WRITE, EMCS_FLUSH, EMCS_CLOSE_OK, EMCS_CLOSE_ERROR, }; struct _write_msg { EMsg msg; enum _write_msg_t op; const char *data; size_t n; }; static void em_camel_stream_class_init (EMCamelStreamClass *klass); static void em_camel_stream_init (CamelObject *object); static void em_camel_stream_finalize (CamelObject *object); static ssize_t stream_write(CamelStream *stream, const char *buffer, size_t n); static int stream_close(CamelStream *stream); static int stream_flush(CamelStream *stream); static CamelStreamClass *parent_class = NULL; CamelType em_camel_stream_get_type (void) { static CamelType type = CAMEL_INVALID_TYPE; if (type == CAMEL_INVALID_TYPE) { type = camel_type_register (CAMEL_STREAM_TYPE, "EMCamelStream", sizeof (EMCamelStream), sizeof (EMCamelStreamClass), (CamelObjectClassInitFunc) em_camel_stream_class_init, NULL, (CamelObjectInitFunc) em_camel_stream_init, (CamelObjectFinalizeFunc) em_camel_stream_finalize); } return type; } static void em_camel_stream_class_init (EMCamelStreamClass *klass) { CamelStreamClass *stream_class = CAMEL_STREAM_CLASS (klass); parent_class = (CamelStreamClass *) CAMEL_STREAM_TYPE; /* virtual method overload */ stream_class->write = stream_write; stream_class->flush = stream_flush; stream_class->close = stream_close; } static gboolean emcs_gui_received(GIOChannel *source, GIOCondition cond, void *data) { EMCamelStream *estream = data; struct _write_msg *msg; d(printf("%p: gui sync op job waiting\n", estream)); msg = (struct _write_msg *)e_msgport_get(estream->data_port); /* Should never happen ... */ if (msg == NULL) return TRUE; d(printf("%p: running sync op %d\n", estream, msg->op)); /* force out any pending data before doing anything else */ if (estream->used > 0) { d(printf("sync write %d\n", estream->used)); if (estream->html_stream) gtk_html_stream_write(estream->html_stream, estream->buffer, estream->used); estream->used = 0; } switch (msg->op) { case EMCS_WRITE: d(printf("sync write %d\n", msg->n)); if (estream->html_stream) gtk_html_stream_write(estream->html_stream, msg->data, msg->n); break; case EMCS_FLUSH: stream_flush((CamelStream *)estream); break; case EMCS_CLOSE_OK: if (estream->html_stream) { gtk_html_stream_close(estream->html_stream, GTK_HTML_STREAM_OK); estream->html_stream = NULL; } break; case EMCS_CLOSE_ERROR: if (estream->html_stream) { gtk_html_stream_close(estream->html_stream, GTK_HTML_STREAM_ERROR); estream->html_stream = NULL; } break; } e_msgport_reply((EMsg *)msg); d(printf("%p: gui sync op jobs done\n", estream)); return TRUE; } static void em_camel_stream_init (CamelObject *object) { EMCamelStream *estream = (EMCamelStream *)object; estream->data_port = e_msgport_new(); estream->reply_port = e_msgport_new(); estream->gui_channel = g_io_channel_unix_new(e_msgport_fd(estream->data_port)); estream->gui_watch = g_io_add_watch(estream->gui_channel, G_IO_IN, emcs_gui_received, estream); estream->used = 0; estream->buffer = g_malloc(EMCS_BUFFER_SIZE); d(printf("%p: new estream\n", estream)); } static void sync_op(EMCamelStream *estream, enum _write_msg_t op, const char *data, size_t n) { struct _write_msg msg; d(printf("%p: launching sync op %d\n", estream, op)); /* we do everything synchronous, we should never have any locks, and this prevents overflow from banked up data */ msg.msg.reply_port = estream->reply_port; msg.op = op; msg.data = data; msg.n = n; e_msgport_put(estream->data_port, &msg.msg); e_msgport_wait(estream->reply_port); g_assert(e_msgport_get(msg.msg.reply_port) == &msg.msg); d(printf("%p: returned sync op %d\n", estream, op)); } static void em_camel_stream_finalize (CamelObject *object) { EMCamelStream *estream = (EMCamelStream *)object; d(printf("%p: finalising stream\n", object)); if (estream->html_stream) { d(printf("%p: html stream still open - error\n", object)); if (pthread_self() == mail_gui_thread) gtk_html_stream_close(estream->html_stream, GTK_HTML_STREAM_ERROR); else sync_op(estream, EMCS_CLOSE_ERROR, NULL, 0); } /* TODO: is this stuff safe to do in another thread? */ g_source_remove(estream->gui_watch); g_io_channel_unref(estream->gui_channel); e_msgport_destroy(estream->data_port); estream->data_port = NULL; e_msgport_destroy(estream->reply_port); estream->reply_port = NULL; g_free(estream->buffer); } static ssize_t stream_write (CamelStream *stream, const char *buffer, size_t n) { EMCamelStream *estream = EM_CAMEL_STREAM (stream); if (estream->html_stream == NULL) return -1; #ifdef LOG_STREAM if (estream->save) fwrite(buffer, sizeof(char), n, estream->save); #endif if (pthread_self() == mail_gui_thread) gtk_html_stream_write(estream->html_stream, buffer, n); else { #if 1 size_t left = EMCS_BUFFER_SIZE-estream->used; /* A super-simple buffer, if we get too much to fit, just do a sync write, which will implicitly clear our previous writes first */ d(printf("thread write '%d'\n", n)); if (n >= left) { sync_op(estream, EMCS_WRITE, buffer, n); } else { memcpy(estream->buffer + estream->used, buffer, n); estream->used += n; } #else sync_op(estream, EMCS_WRITE, buffer, n); #endif } return (ssize_t) n; } static int stream_flush(CamelStream *stream) { EMCamelStream *estream = (EMCamelStream *)stream; if (estream->html_stream) { if (pthread_self() == mail_gui_thread) { /* FIXME: flush html stream via gtkhtml_stream_flush which doens't exist yet ... */ while (gtk_events_pending ()) gtk_main_iteration (); } else { sync_op(estream, EMCS_FLUSH, NULL, 0); } } return 0; } static int stream_close(CamelStream *stream) { EMCamelStream *estream = (EMCamelStream *)stream; d(printf("%p: closing stream\n", stream)); #ifdef LOG_STREAM if (estream->save) { fclose(estream->save); estream->save = NULL; } #endif if (estream->html_stream) { if (pthread_self() == mail_gui_thread) { gtk_html_stream_close(estream->html_stream, GTK_HTML_STREAM_OK); estream->html_stream = NULL; } else { sync_op(estream, EMCS_CLOSE_OK, NULL, 0); } } return 0; } static void emcs_gtkhtml_destroy(struct _GtkHTML *html, EMCamelStream *emcs) { d(printf("%p: emcs gtkhtml destroy\n", emcs)); emcs->html = NULL; emcs->html_stream = NULL; } /* TODO: Could pass NULL for html_stream, and do a gtk_html_begin on first data -> less flashing */ CamelStream * em_camel_stream_new(struct _GtkHTML *html, struct _GtkHTMLStream *html_stream) { EMCamelStream *new; new = EM_CAMEL_STREAM (camel_object_new (EM_CAMEL_STREAM_TYPE)); new->html_stream = html_stream; g_signal_connect(html, "destroy", G_CALLBACK(emcs_gtkhtml_destroy), new); #ifdef LOG_STREAM { static int count; char name[32]; sprintf(name, "camel-stream.%d.html", count++); printf("saving raw html to '%s'\n", name); new->save = fopen(name, "w"); } #endif return CAMEL_STREAM (new); }