LCOV - code coverage report
Current view: top level - src - reader.c (source / functions) Hit Total Coverage
Test: coverage.info Lines: 91 176 51.7 %
Date: 2016-09-12 12:20:35 Functions: 8 12 66.7 %

          Line data    Source code
       1             : /* reader.c - provides the Reader object
       2             :  * Copyright (C) 2001, 2010, 2012 g10 Code GmbH
       3             :  *
       4             :  * This file is part of KSBA.
       5             :  *
       6             :  * KSBA is free software; you can redistribute it and/or modify
       7             :  * it under the terms of either
       8             :  *
       9             :  *   - the GNU Lesser General Public License as published by the Free
      10             :  *     Software Foundation; either version 3 of the License, or (at
      11             :  *     your option) any later version.
      12             :  *
      13             :  * or
      14             :  *
      15             :  *   - the GNU General Public License as published by the Free
      16             :  *     Software Foundation; either version 2 of the License, or (at
      17             :  *     your option) any later version.
      18             :  *
      19             :  * or both in parallel, as here.
      20             :  *
      21             :  * KSBA is distributed in the hope that it will be useful, but WITHOUT
      22             :  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
      23             :  * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public
      24             :  * License for more details.
      25             :  *
      26             :  * You should have received a copies of the GNU General Public License
      27             :  * and the GNU Lesser General Public License along with this program;
      28             :  * if not, see <http://www.gnu.org/licenses/>.
      29             :  */
      30             : 
      31             : #include <config.h>
      32             : #include <stdio.h>
      33             : #include <stdlib.h>
      34             : #include <string.h>
      35             : #include <assert.h>
      36             : #include <errno.h>
      37             : #include "util.h"
      38             : 
      39             : #include "ksba.h"
      40             : #include "reader.h"
      41             : 
      42             : /**
      43             :  * ksba_reader_new:
      44             :  *
      45             :  * Create a new but uninitialized ksba_reader_t Object.  Using this
      46             :  * reader object in unitialized state does always yield eof.
      47             :  *
      48             :  * Return value: ksba_reader_t object or an error code.
      49             :  **/
      50             : gpg_error_t
      51           6 : ksba_reader_new (ksba_reader_t *r_r)
      52             : {
      53           6 :   *r_r = xtrycalloc (1, sizeof **r_r);
      54           6 :   if (!*r_r)
      55           0 :     return gpg_error_from_errno (errno);
      56           6 :   return 0;
      57             : }
      58             : 
      59             : 
      60             : /**
      61             :  * ksba_reader_release:
      62             :  * @r: Reader Object (or NULL)
      63             :  *
      64             :  * Release this object
      65             :  **/
      66             : void
      67           6 : ksba_reader_release (ksba_reader_t r)
      68             : {
      69           6 :   if (!r)
      70           6 :     return;
      71           6 :   if (r->notify_cb)
      72             :     {
      73           0 :       void (*notify_fnc)(void*,ksba_reader_t) = r->notify_cb;
      74             : 
      75           0 :       r->notify_cb = NULL;
      76           0 :       notify_fnc (r->notify_cb_value, r);
      77             :     }
      78           6 :   if (r->type == READER_TYPE_MEM)
      79           2 :     xfree (r->u.mem.buffer);
      80           6 :   xfree (r->unread.buf);
      81           6 :   xfree (r);
      82             : }
      83             : 
      84             : 
      85             : /* Set NOTIFY as function to be called by ksba_reader_release before
      86             :    resources are actually deallocated.  NOTIFY_VALUE is passed to the
      87             :    called function as its first argument.  Note that only the last
      88             :    registered function will be called; passing NULL for NOTIFY removes
      89             :    the notification.  */
      90             : gpg_error_t
      91           0 : ksba_reader_set_release_notify (ksba_reader_t r,
      92             :                                 void (*notify)(void*,ksba_reader_t),
      93             :                                 void *notify_value)
      94             : {
      95           0 :   if (!r)
      96           0 :     return gpg_error (GPG_ERR_INV_VALUE);
      97           0 :   r->notify_cb = notify;
      98           0 :   r->notify_cb_value = notify_value;
      99           0 :   return 0;
     100             : }
     101             : 
     102             : 
     103             : /* Clear the error and eof indicators for READER, so that it can be
     104             :    continued to use.  Also dicards any unread bytes. This is usually
     105             :    required if the upper layer want to send to send an EOF to indicate
     106             :    the logical end of one part of a file.  If BUFFER and BUFLEN are
     107             :    not NULL, possible unread data is copied to a newly allocated
     108             :    buffer and this buffer is assigned to BUFFER, BUFLEN will be set to
     109             :    the length of the unread bytes. */
     110             : gpg_error_t
     111           0 : ksba_reader_clear (ksba_reader_t r, unsigned char **buffer, size_t *buflen)
     112             : {
     113             :   size_t n;
     114             : 
     115           0 :   if (!r)
     116           0 :     return gpg_error (GPG_ERR_INV_VALUE);
     117             : 
     118           0 :   r->eof = 0;
     119           0 :   r->error = 0;
     120           0 :   r->nread = 0;
     121           0 :   n = r->unread.length;
     122           0 :   r->unread.length = 0;
     123             : 
     124           0 :   if (buffer && buflen)
     125             :     {
     126           0 :       *buffer = NULL;
     127           0 :       *buflen = 0;
     128           0 :       if (n)
     129             :         {
     130           0 :           *buffer = xtrymalloc (n);
     131           0 :           if (!*buffer)
     132           0 :             return gpg_error_from_errno (errno);
     133           0 :           memcpy (*buffer, r->unread.buf, n);
     134           0 :           *buflen = n;
     135             :         }
     136             :     }
     137             : 
     138           0 :   return 0;
     139             : }
     140             : 
     141             : 
     142             : gpg_error_t
     143           8 : ksba_reader_error (ksba_reader_t r)
     144             : {
     145           8 :   return r? gpg_error_from_errno (r->error) : gpg_error (GPG_ERR_INV_VALUE);
     146             : }
     147             : 
     148             : unsigned long
     149         312 : ksba_reader_tell (ksba_reader_t r)
     150             : {
     151         312 :   return r? r->nread : 0;
     152             : }
     153             : 
     154             : 
     155             : /**
     156             :  * ksba_reader_set_mem:
     157             :  * @r: Reader object
     158             :  * @buffer: Data
     159             :  * @length: Length of Data (bytes)
     160             :  *
     161             :  * Intialize the reader object with @length bytes from @buffer and set
     162             :  * the read position to the beginning.  It is possible to reuse this
     163             :  * reader object with another buffer if the reader object has
     164             :  * already been initialized using this function.
     165             :  *
     166             :  * Return value: 0 on success or an error code.
     167             :  **/
     168             : gpg_error_t
     169           2 : ksba_reader_set_mem (ksba_reader_t r, const void *buffer, size_t length)
     170             : {
     171           2 :   if (!r || !buffer)
     172           0 :     return gpg_error (GPG_ERR_INV_VALUE);
     173           2 :   if (r->type == READER_TYPE_MEM)
     174             :     { /* Reuse this reader */
     175           0 :       xfree (r->u.mem.buffer);
     176           0 :       r->type = 0;
     177             :     }
     178           2 :   if (r->type)
     179           0 :     return gpg_error (GPG_ERR_CONFLICT);
     180             : 
     181           2 :   r->u.mem.buffer = xtrymalloc (length);
     182           2 :   if (!r->u.mem.buffer)
     183           0 :     return gpg_error (GPG_ERR_ENOMEM);
     184           2 :   memcpy (r->u.mem.buffer, buffer, length);
     185           2 :   r->u.mem.size = length;
     186           2 :   r->u.mem.readpos = 0;
     187           2 :   r->type = READER_TYPE_MEM;
     188           2 :   r->eof = 0;
     189             : 
     190           2 :   return 0;
     191             : }
     192             : 
     193             : 
     194             : /**
     195             :  * ksba_reader_set_fd:
     196             :  * @r: Reader object
     197             :  * @fd: file descriptor
     198             :  *
     199             :  * Initialize the Reader object with a file descriptor, so that read
     200             :  * operations on this object are excuted on this file descriptor.
     201             :  *
     202             :  * Return value:
     203             :  **/
     204             : gpg_error_t
     205           0 : ksba_reader_set_fd (ksba_reader_t r, int fd)
     206             : {
     207           0 :   if (!r || fd == -1)
     208           0 :     return gpg_error (GPG_ERR_INV_VALUE);
     209           0 :   if (r->type)
     210           0 :     return gpg_error (GPG_ERR_CONFLICT);
     211             : 
     212           0 :   r->eof = 0;
     213           0 :   r->type = READER_TYPE_FD;
     214           0 :   r->u.fd = fd;
     215             : 
     216           0 :   return 0;
     217             : }
     218             : 
     219             : /**
     220             :  * ksba_reader_set_file:
     221             :  * @r: Reader object
     222             :  * @fp: file pointer
     223             :  *
     224             :  * Initialize the Reader object with a stdio file pointer, so that read
     225             :  * operations on this object are excuted on this stream
     226             :  *
     227             :  * Return value:
     228             :  **/
     229             : gpg_error_t
     230           4 : ksba_reader_set_file (ksba_reader_t r, FILE *fp)
     231             : {
     232           4 :   if (!r || !fp)
     233           0 :     return gpg_error (GPG_ERR_INV_VALUE);
     234           4 :   if (r->type)
     235           0 :     return gpg_error (GPG_ERR_CONFLICT);
     236             : 
     237           4 :   r->eof = 0;
     238           4 :   r->type = READER_TYPE_FILE;
     239           4 :   r->u.file = fp;
     240           4 :   return 0;
     241             : }
     242             : 
     243             : 
     244             : 
     245             : /**
     246             :  * ksba_reader_set_cb:
     247             :  * @r: Reader object
     248             :  * @cb: Callback function
     249             :  * @cb_value: Value passed to the callback function
     250             :  *
     251             :  * Initialize the reader object with a callback function.
     252             :  * This callback function is defined as:
     253             :  * <literal>
     254             :  * typedef int (*cb) (void *cb_value,
     255             :  *                    char *buffer, size_t count,
     256             :  *                    size_t *nread);
     257             :  * </literal>
     258             :  *
     259             :  * The callback should return a maximium of @count bytes in @buffer
     260             :  * and the number actually read in @nread.  It may return 0 in @nread
     261             :  * if there are no bytes currently available.  To indicate EOF the
     262             :  * callback should return with an error code of GPG_ERR_EOF and set @nread to
     263             :  * 0.  The callback may support passing %NULL for @buffer and @nread
     264             :  * and %0 for count as an indication to reset its internal read
     265             :  * pointer.
     266             :  *
     267             :  * Return value: 0 on success or an error code
     268             :  **/
     269             : gpg_error_t
     270           0 : ksba_reader_set_cb (ksba_reader_t r,
     271             :                     int (*cb)(void*,char *,size_t,size_t*), void *cb_value )
     272             : {
     273           0 :   if (!r || !cb)
     274           0 :     return gpg_error (GPG_ERR_INV_VALUE);
     275           0 :   if (r->type)
     276           0 :     return gpg_error (GPG_ERR_CONFLICT);
     277             : 
     278           0 :   r->eof = 0;
     279           0 :   r->type = READER_TYPE_CB;
     280           0 :   r->u.cb.fnc = cb;
     281           0 :   r->u.cb.value = cb_value;
     282             : 
     283           0 :   return 0;
     284             : }
     285             : 
     286             : 
     287             : /**
     288             :  * ksba_reader_read:
     289             :  * @r: Readder object
     290             :  * @buffer: A buffer for returning the data
     291             :  * @length: The length of this buffer
     292             :  * @nread:  Number of bytes actually read.
     293             :  *
     294             :  * Read data from the current read position to the supplied @buffer,
     295             :  * max. @length bytes are read and the actual number of bytes read are
     296             :  * returned in @nread.  If there are no more bytes available %GPG_ERR_EOF is
     297             :  * returned and @nread is set to 0.
     298             :  *
     299             :  * If a @buffer of NULL is specified, the function does only return
     300             :  * the number of bytes available and does not move the read pointer.
     301             :  * This does only work for objects initialized from memory; if the
     302             :  * object is not capable of this it will return the error
     303             :  * GPG_ERR_NOT_IMPLEMENTED
     304             :  *
     305             :  * Return value: 0 on success, GPG_ERR_EOF or another error code
     306             :  **/
     307             : gpg_error_t
     308         984 : ksba_reader_read (ksba_reader_t r, char *buffer, size_t length, size_t *nread)
     309             : {
     310             :   size_t nbytes;
     311             : 
     312         984 :   if (!r || !nread)
     313           0 :     return gpg_error (GPG_ERR_INV_VALUE);
     314             : 
     315             : 
     316         984 :   if (!buffer)
     317             :     {
     318           0 :       if (r->type != READER_TYPE_MEM)
     319           0 :         return gpg_error (GPG_ERR_NOT_IMPLEMENTED);
     320           0 :       *nread = r->u.mem.size - r->u.mem.readpos;
     321           0 :       if (r->unread.buf)
     322           0 :         *nread += r->unread.length - r->unread.readpos;
     323           0 :       return *nread? 0 : gpg_error (GPG_ERR_EOF);
     324             :     }
     325             : 
     326         984 :   *nread = 0;
     327             : 
     328         984 :   if (r->unread.buf && r->unread.length)
     329             :     {
     330           2 :       nbytes = r->unread.length - r->unread.readpos;
     331           2 :       if (!nbytes)
     332           0 :         return gpg_error (GPG_ERR_BUG);
     333             : 
     334           2 :       if (nbytes > length)
     335           1 :         nbytes = length;
     336           2 :       memcpy (buffer, r->unread.buf + r->unread.readpos, nbytes);
     337           2 :       r->unread.readpos += nbytes;
     338           2 :       if (r->unread.readpos == r->unread.length)
     339           1 :         r->unread.readpos = r->unread.length = 0;
     340           2 :       *nread = nbytes;
     341           2 :       r->nread += nbytes;
     342           2 :       return 0;
     343             :     }
     344             : 
     345             : 
     346         982 :   if (!r->type)
     347             :     {
     348           0 :       r->eof = 1;
     349           0 :       return gpg_error (GPG_ERR_EOF);
     350             :     }
     351         982 :   else if (r->type == READER_TYPE_MEM)
     352             :     {
     353          86 :       nbytes = r->u.mem.size - r->u.mem.readpos;
     354          86 :       if (!nbytes)
     355             :         {
     356           2 :           r->eof = 1;
     357           2 :           return gpg_error (GPG_ERR_EOF);
     358             :         }
     359             : 
     360          84 :       if (nbytes > length)
     361          82 :         nbytes = length;
     362          84 :       memcpy (buffer, r->u.mem.buffer + r->u.mem.readpos, nbytes);
     363          84 :       *nread = nbytes;
     364          84 :       r->nread += nbytes;
     365          84 :       r->u.mem.readpos += nbytes;
     366             :     }
     367         896 :   else if (r->type == READER_TYPE_FILE)
     368             :     {
     369             :       size_t n;
     370             : 
     371         896 :       if (r->eof)
     372           3 :         return gpg_error (GPG_ERR_EOF);
     373             : 
     374         893 :       if (!length)
     375             :         {
     376           0 :           *nread = 0;
     377           0 :           return 0;
     378             :         }
     379             : 
     380         893 :       n = fread (buffer, 1, length, r->u.file);
     381         893 :       if (n)
     382             :         {
     383         890 :           r->nread += n;
     384         890 :           *nread = n;
     385             :         }
     386             :       else
     387           3 :         *nread = 0;
     388         893 :       if (n < length)
     389             :         {
     390           3 :           if (ferror(r->u.file))
     391           0 :             r->error = errno;
     392           3 :           r->eof = 1;
     393           3 :           if (!n)
     394           3 :             return gpg_error (GPG_ERR_EOF);
     395             :         }
     396             :     }
     397           0 :   else if (r->type == READER_TYPE_CB)
     398             :     {
     399           0 :       if (r->eof)
     400           0 :         return gpg_error (GPG_ERR_EOF);
     401             : 
     402           0 :       if (r->u.cb.fnc (r->u.cb.value, buffer, length, nread))
     403             :         {
     404           0 :           *nread = 0;
     405           0 :           r->eof = 1;
     406           0 :           return gpg_error (GPG_ERR_EOF);
     407             :         }
     408           0 :       r->nread += *nread;
     409             :     }
     410             :   else
     411           0 :     return gpg_error (GPG_ERR_BUG);
     412             : 
     413         974 :   return 0;
     414             : }
     415             : 
     416             : gpg_error_t
     417           1 : ksba_reader_unread (ksba_reader_t r, const void *buffer, size_t count)
     418             : {
     419           1 :   if (!r || !buffer)
     420           0 :     return gpg_error (GPG_ERR_INV_VALUE);
     421           1 :   if (!count)
     422           0 :     return 0;
     423             : 
     424             :   /* Make sure that we do not push more bytes back than we have read.
     425             :      Otherwise r->nread won't have a clear semantic. */
     426           1 :   if (r->nread < count)
     427           0 :     return gpg_error (GPG_ERR_CONFLICT);
     428             : 
     429           1 :   if (!r->unread.buf)
     430             :     {
     431           1 :       r->unread.size = count + 100;
     432           1 :       r->unread.buf = xtrymalloc (r->unread.size);
     433           1 :       if (!r->unread.buf)
     434           0 :         return gpg_error (GPG_ERR_ENOMEM);
     435           1 :       r->unread.length = count;
     436           1 :       r->unread.readpos = 0;
     437           1 :       memcpy (r->unread.buf, buffer, count);
     438           1 :       r->nread -= count;
     439             :     }
     440           0 :   else if (r->unread.length + count < r->unread.size)
     441             :     {
     442           0 :       memcpy (r->unread.buf+r->unread.length, buffer, count);
     443           0 :       r->unread.length += count;
     444           0 :       r->nread -= count;
     445             :     }
     446             :   else
     447           0 :     return gpg_error (GPG_ERR_NOT_IMPLEMENTED); /* fixme: easy to do */
     448             : 
     449           1 :   return 0;
     450             : }

Generated by: LCOV version 1.11