34a18d46a3df243c4cb2ce1cf601ece9cfb4d499
[gpgme.git] / src / data-stream.c
1 /* data-stream.c - A stream based data object.
2    Copyright (C) 2002, 2004 g10 Code GmbH
3  
4    This file is part of GPGME.
5  
6    GPGME is free software; you can redistribute it and/or modify it
7    under the terms of the GNU Lesser General Public License as
8    published by the Free Software Foundation; either version 2.1 of
9    the License, or (at your option) any later version.
10    
11    GPGME is distributed in the hope that it will be useful, but
12    WITHOUT ANY WARRANTY; without even the implied warranty of
13    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14    Lesser General Public License for more details.
15    
16    You should have received a copy of the GNU Lesser General Public
17    License along with this program; if not, write to the Free Software
18    Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
19    02111-1307, USA.  */
20
21 #if HAVE_CONFIG_H
22 #include <config.h>
23 #endif
24
25 #include <stdio.h>
26 #include <sys/types.h>
27
28 #include "debug.h"
29 #include "data.h"
30
31 \f
32 static ssize_t
33 stream_read (gpgme_data_t dh, void *buffer, size_t size)
34 {
35   size_t amt = fread (buffer, 1, size, dh->data.stream);
36   if (amt > 0)
37     return amt;
38   return ferror (dh->data.stream) ? -1 : 0;
39 }
40
41
42 static ssize_t
43 stream_write (gpgme_data_t dh, const void *buffer, size_t size)
44 {
45   size_t amt = fwrite (buffer, 1, size, dh->data.stream);
46   if (amt > 0)
47     return amt;
48   return ferror (dh->data.stream) ? -1 : 0;
49 }
50
51
52 static off_t
53 stream_seek (gpgme_data_t dh, off_t offset, int whence)
54 {
55   int err;
56
57 #ifdef HAVE_FSEEKO
58   err = fseeko (dh->data.stream, offset, whence);
59 #else
60   /* FIXME: Check for overflow, or at least bail at compilation.  */
61   err = fseek (dh->data.stream, offset, whence);
62 #endif
63
64   if (err)
65     return -1;
66
67 #ifdef HAVE_FSEEKO
68   return ftello (dh->data.stream);
69 #else
70   return ftell (dh->data.stream);
71 #endif
72 }
73
74
75 static int
76 stream_get_fd (gpgme_data_t dh)
77 {
78   fflush (dh->data.stream);
79   return fileno (dh->data.stream);
80 }
81
82
83 static struct _gpgme_data_cbs stream_cbs =
84   {
85     stream_read,
86     stream_write,
87     stream_seek,
88     NULL,
89     stream_get_fd
90   };
91
92 \f
93 gpgme_error_t
94 gpgme_data_new_from_stream (gpgme_data_t *r_dh, FILE *stream)
95 {
96   gpgme_error_t err;
97   TRACE_BEG1 (DEBUG_DATA, "gpgme_data_new_from_stream", r_dh, "stream=%p",
98               stream);
99
100   err = _gpgme_data_new (r_dh, &stream_cbs);
101   if (err)
102     return TRACE_ERR (err);
103
104   (*r_dh)->data.stream = stream;
105   return TRACE_SUC1 ("dh=%p", *r_dh);
106 }