2002-03-25 23:16:26 +03:00
# ifndef H_RPMIO_INTERNAL
# define H_RPMIO_INTERNAL
/** \ingroup rpmio
* \ file rpmio / rpmio_internal . h
*/
# include <rpmio.h>
# include <rpmurl.h>
# include <rpmpgp.h>
/** \ingroup rpmio
*/
typedef struct _FDSTACK_s {
FDIO_t io ;
/*@dependent@*/ void * fp ;
int fdno ;
} FDSTACK_t ;
/** \ingroup rpmio
* Cumulative statistics for an I / O operation .
*/
typedef struct {
int count ; /*!< Number of operations. */
off_t bytes ; /*!< Number of bytes transferred. */
time_t msecs ; /*!< Number of milli-seconds. */
} OPSTAT_t ;
/** \ingroup rpmio
* Identify per - desciptor I / O operation statistics .
*/
enum FDSTAT_e {
FDSTAT_READ = 0 , /*!< Read statistics index. */
FDSTAT_WRITE = 1 , /*!< Write statistics index. */
FDSTAT_SEEK = 2 , /*!< Seek statistics index. */
FDSTAT_CLOSE = 3 /*!< Close statistics index */
} ;
/** \ingroup rpmio
* Cumulative statistics for a descriptor .
*/
typedef /*@abstract@*/ struct {
struct timeval create ; /*!< Structure creation time. */
struct timeval begin ; /*!< Operation start time. */
OPSTAT_t ops [ 4 ] ; /*!< Cumulative statistics. */
} * FDSTAT_t ;
/** \ingroup rpmio
* Bit ( s ) to control digest operation .
*/
typedef enum rpmDigestFlags_e {
RPMDIGEST_NONE = 0
} rpmDigestFlags ;
/** \ingroup rpmio
*/
typedef struct _FDDIGEST_s {
pgpHashAlgo hashalgo ;
DIGEST_CTX hashctx ;
} * FDDIGEST_t ;
/** \ingroup rpmio
* Duplicate a digest context .
* @ param octx existing digest context
* @ return duplicated digest context
*/
2002-08-03 20:35:14 +04:00
/*@only@*/
2002-03-25 23:16:26 +03:00
DIGEST_CTX rpmDigestDup ( DIGEST_CTX octx )
/*@*/ ;
/** \ingroup rpmio
* Initialize digest .
* Set bit count to 0 and buffer to mysterious initialization constants .
2002-08-03 20:35:14 +04:00
* @ param hashalgo type of digest
2002-03-25 23:16:26 +03:00
* @ param flags bit ( s ) to control digest operation
* @ return digest context
*/
/*@only@*/
DIGEST_CTX rpmDigestInit ( pgpHashAlgo hashalgo , rpmDigestFlags flags )
/*@*/ ;
/** \ingroup rpmio
* Update context with next plain text buffer .
* @ param ctx digest context
* @ param data next data buffer
* @ param len no . bytes of data
* @ return 0 on success
*/
int rpmDigestUpdate ( DIGEST_CTX ctx , const void * data , size_t len )
/*@modifies ctx @*/ ;
/** \ingroup rpmio
* Return digest and destroy context .
* Final wrapup - pad to 64 - byte boundary with the bit pattern
* 1 0 * ( 64 - bit count of bits processed , MSB - first )
*
* @ param ctx digest context
* @ retval datap address of returned digest
* @ retval lenp address of digest length
* @ param asAscii return digest as ascii string ?
* @ return 0 on success
*/
int rpmDigestFinal ( /*@only@*/ DIGEST_CTX ctx ,
/*@null@*/ /*@out@*/ void * * datap ,
/*@null@*/ /*@out@*/ size_t * lenp , int asAscii )
/*@modifies *datap, *lenp @*/ ;
/** \ingroup rpmio
* The FD_t File Handle data structure .
*/
struct _FD_s {
/*@refs@*/ int nrefs ;
int flags ;
# define RPMIO_DEBUG_IO 0x40000000
# define RPMIO_DEBUG_REFS 0x20000000
int magic ;
# define FDMAGIC 0x04463138
int nfps ;
FDSTACK_t fps [ 8 ] ;
int urlType ; /* ufdio: */
/*@dependent@*/ void * url ; /* ufdio: URL info */
int rd_timeoutsecs ; /* ufdRead: per FD_t timer */
ssize_t bytesRemain ; /* ufdio: */
ssize_t contentLength ; /* ufdio: */
int persist ; /* ufdio: */
int wr_chunked ; /* ufdio: */
int syserrno ; /* last system errno encountered */
/*@observer@*/ const void * errcookie ; /* gzdio/bzdio/ufdio: */
FDSTAT_t stats ; /* I/O statistics */
int ndigests ;
# define FDDIGEST_MAX 4
struct _FDDIGEST_s digests [ FDDIGEST_MAX ] ;
int ftpFileDoneNeeded ; /* ufdio: (FTP) */
unsigned int firstFree ; /* fadio: */
long int fileSize ; /* fadio: */
long int fd_cpioPos ; /* cpio: */
} ;
/*@access FD_t@*/
# define FDSANE(fd) assert(fd && fd->magic == FDMAGIC)
/*@-redecl@*/
/*@unchecked@*/
extern int _rpmio_debug ;
/*@=redecl@*/
2002-08-03 20:35:14 +04:00
/*@-redecl@*/
extern int _ftp_debug ;
/*@=redecl@*/
2002-03-25 23:16:26 +03:00
# define DBG(_f, _m, _x) \
if ( ( _rpmio_debug | ( ( _f ) ? ( ( FD_t ) ( _f ) ) - > flags : 0 ) ) & ( _m ) ) fprintf _x
2002-08-03 20:35:14 +04:00
# if defined(__LCLINT__XXX)
# define DBGIO(_f, _x)
# define DBGREFS(_f, _x)
# else
2002-03-25 23:16:26 +03:00
# define DBGIO(_f, _x) DBG((_f), RPMIO_DEBUG_IO, _x)
# define DBGREFS(_f, _x) DBG((_f), RPMIO_DEBUG_REFS, _x)
2002-08-03 20:35:14 +04:00
# endif
2002-03-25 23:16:26 +03:00
# ifdef __cplusplus
extern " C " {
# endif
/** \ingroup rpmio
*/
int fdFgets ( FD_t fd , char * buf , size_t len )
/*@globals errno, fileSystem @*/
/*@modifies *buf, fd, errno, fileSystem @*/ ;
/** \ingroup rpmio
*/
/*@null@*/ FD_t ftpOpen ( const char * url , /*@unused@*/ int flags ,
/*@unused@*/ mode_t mode , /*@out@*/ urlinfo * uret )
/*@globals fileSystem @*/
/*@modifies *uret, fileSystem @*/ ;
/** \ingroup rpmio
*/
int ftpReq ( FD_t data , const char * ftpCmd , const char * ftpArg )
/*@globals fileSystem @*/
/*@modifies data, fileSystem @*/ ;
/** \ingroup rpmio
*/
int ftpCmd ( const char * cmd , const char * url , const char * arg2 )
/*@globals fileSystem @*/
/*@modifies fileSystem @*/ ;
/** \ingroup rpmio
*/
int ufdClose ( /*@only@*/ void * cookie )
/*@globals fileSystem @*/
/*@modifies cookie, fileSystem @*/ ;
/** \ingroup rpmio
*/
/*@unused@*/ static inline
2002-06-05 15:06:46 +04:00
/*@null@*/ FDIO_t fdGetIo ( FD_t fd )
2002-03-25 23:16:26 +03:00
/*@*/
{
FDSANE ( fd ) ;
return fd - > fps [ fd - > nfps ] . io ;
}
/** \ingroup rpmio
*/
/*@-nullstate@*/ /* FIX: io may be NULL */
/*@unused@*/ static inline
void fdSetIo ( FD_t fd , /*@kept@*/ /*@null@*/ FDIO_t io )
/*@modifies fd @*/
{
FDSANE ( fd ) ;
/*@-assignexpose@*/
fd - > fps [ fd - > nfps ] . io = io ;
/*@=assignexpose@*/
}
/*@=nullstate@*/
/** \ingroup rpmio
*/
/*@unused@*/ static inline
/*@exposed@*/ /*@dependent@*/ /*@null@*/ FILE * fdGetFILE ( FD_t fd )
/*@*/
{
FDSANE ( fd ) ;
/*@+voidabstract@*/
return ( ( FILE * ) fd - > fps [ fd - > nfps ] . fp ) ;
/*@=voidabstract@*/
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline
/*@exposed@*/ /*@dependent@*/ /*@null@*/ void * fdGetFp ( FD_t fd )
/*@*/
{
FDSANE ( fd ) ;
return fd - > fps [ fd - > nfps ] . fp ;
}
/** \ingroup rpmio
*/
/*@-nullstate@*/ /* FIX: fp may be NULL */
/*@unused@*/ static inline
void fdSetFp ( FD_t fd , /*@kept@*/ /*@null@*/ void * fp )
/*@modifies fd @*/
{
FDSANE ( fd ) ;
/*@-assignexpose@*/
fd - > fps [ fd - > nfps ] . fp = fp ;
/*@=assignexpose@*/
}
/*@=nullstate@*/
/** \ingroup rpmio
*/
/*@unused@*/ static inline
int fdGetFdno ( FD_t fd )
/*@*/
{
FDSANE ( fd ) ;
return fd - > fps [ fd - > nfps ] . fdno ;
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline
void fdSetFdno ( FD_t fd , int fdno )
/*@modifies fd @*/
{
FDSANE ( fd ) ;
fd - > fps [ fd - > nfps ] . fdno = fdno ;
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline
void fdSetContentLength ( FD_t fd , ssize_t contentLength )
/*@modifies fd @*/
{
FDSANE ( fd ) ;
fd - > contentLength = fd - > bytesRemain = contentLength ;
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline
void fdPush ( FD_t fd , FDIO_t io , void * fp , int fdno )
/*@modifies fd @*/
{
FDSANE ( fd ) ;
if ( fd - > nfps > = ( sizeof ( fd - > fps ) / sizeof ( fd - > fps [ 0 ] ) - 1 ) )
return ;
fd - > nfps + + ;
fdSetIo ( fd , io ) ;
fdSetFp ( fd , fp ) ;
fdSetFdno ( fd , fdno ) ;
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline void fdPop ( FD_t fd )
/*@modifies fd @*/
{
FDSANE ( fd ) ;
if ( fd - > nfps < 0 ) return ;
fdSetIo ( fd , NULL ) ;
fdSetFp ( fd , NULL ) ;
fdSetFdno ( fd , - 1 ) ;
fd - > nfps - - ;
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline void fdstat_enter ( /*@null@*/ FD_t fd , int opx )
/*@modifies fd @*/
{
if ( fd = = NULL | | fd - > stats = = NULL ) return ;
fd - > stats - > ops [ opx ] . count + + ;
( void ) gettimeofday ( & fd - > stats - > begin , NULL ) ;
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline
time_t tvsub ( /*@null@*/ const struct timeval * etv ,
/*@null@*/ const struct timeval * btv )
/*@*/
{
time_t secs , usecs ;
if ( etv = = NULL | | btv = = NULL ) return 0 ;
secs = etv - > tv_sec - btv - > tv_sec ;
for ( usecs = etv - > tv_usec - btv - > tv_usec ; usecs < 0 ; usecs + = 1000000 )
secs + + ;
return ( ( secs * 1000 ) + ( usecs / 1000 ) ) ;
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline
void fdstat_exit ( /*@null@*/ FD_t fd , int opx , ssize_t rc )
/*@modifies fd @*/
{
struct timeval end ;
if ( fd = = NULL ) return ;
if ( rc = = - 1 ) fd - > syserrno = errno ;
if ( fd - > stats = = NULL ) return ;
( void ) gettimeofday ( & end , NULL ) ;
if ( rc > = 0 ) {
switch ( opx ) {
case FDSTAT_SEEK :
fd - > stats - > ops [ opx ] . bytes = rc ;
break ;
default :
fd - > stats - > ops [ opx ] . bytes + = rc ;
if ( fd - > bytesRemain > 0 ) fd - > bytesRemain - = rc ;
break ;
}
}
fd - > stats - > ops [ opx ] . msecs + = tvsub ( & end , & fd - > stats - > begin ) ;
fd - > stats - > begin = end ; /* structure assignment */
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline
void fdstat_print ( /*@null@*/ FD_t fd , const char * msg , FILE * fp )
/*@globals fileSystem @*/
/*@modifies *fp, fileSystem @*/
{
int opx ;
if ( fd = = NULL | | fd - > stats = = NULL ) return ;
for ( opx = 0 ; opx < 4 ; opx + + ) {
OPSTAT_t * ops = & fd - > stats - > ops [ opx ] ;
if ( ops - > count < = 0 ) continue ;
switch ( opx ) {
case FDSTAT_READ :
if ( msg ) fprintf ( fp , " %s: " , msg ) ;
fprintf ( fp , " %8d reads, %8ld total bytes in %d.%03d secs \n " ,
ops - > count , ( long ) ops - > bytes ,
( int ) ( ops - > msecs / 1000 ) , ( int ) ( ops - > msecs % 1000 ) ) ;
/*@switchbreak@*/ break ;
case FDSTAT_WRITE :
if ( msg ) fprintf ( fp , " %s: " , msg ) ;
fprintf ( fp , " %8d writes, %8ld total bytes in %d.%03d secs \n " ,
ops - > count , ( long ) ops - > bytes ,
( int ) ( ops - > msecs / 1000 ) , ( int ) ( ops - > msecs % 1000 ) ) ;
/*@switchbreak@*/ break ;
case FDSTAT_SEEK :
/*@switchbreak@*/ break ;
case FDSTAT_CLOSE :
/*@switchbreak@*/ break ;
}
}
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline
void fdSetSyserrno ( FD_t fd , int syserrno , /*@kept@*/ const void * errcookie )
/*@modifies fd @*/
{
FDSANE ( fd ) ;
fd - > syserrno = syserrno ;
/*@-assignexpose@*/
fd - > errcookie = errcookie ;
/*@=assignexpose@*/
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline
int fdGetRdTimeoutSecs ( FD_t fd )
/*@*/
{
FDSANE ( fd ) ;
return fd - > rd_timeoutsecs ;
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline
long int fdGetCpioPos ( FD_t fd )
/*@*/
{
FDSANE ( fd ) ;
return fd - > fd_cpioPos ;
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline
void fdSetCpioPos ( FD_t fd , long int cpioPos )
/*@modifies fd @*/
{
FDSANE ( fd ) ;
fd - > fd_cpioPos = cpioPos ;
}
/** \ingroup rpmio
*/
/*@mayexit@*/ /*@unused@*/ static inline
FD_t c2f ( /*@null@*/ void * cookie )
/*@*/
{
/*@-castexpose@*/
FD_t fd = ( FD_t ) cookie ;
/*@=castexpose@*/
FDSANE ( fd ) ;
/*@-refcounttrans -retalias@*/ return fd ; /*@=refcounttrans =retalias@*/
}
/** \ingroup rpmio
* Attach digest to fd .
*/
/*@unused@*/ static inline
void fdInitDigest ( FD_t fd , pgpHashAlgo hashalgo , int flags )
/*@modifies fd @*/
{
FDDIGEST_t fddig = fd - > digests + fd - > ndigests ;
if ( fddig ! = ( fd - > digests + FDDIGEST_MAX ) ) {
fd - > ndigests + + ;
fddig - > hashalgo = hashalgo ;
fddig - > hashctx = rpmDigestInit ( hashalgo , flags ) ;
}
}
/** \ingroup rpmio
* Update digest ( s ) attached to fd .
*/
/*@unused@*/ static inline
2002-08-03 20:35:14 +04:00
void fdUpdateDigests ( FD_t fd , const unsigned char * buf , ssize_t buflen )
2002-03-25 23:16:26 +03:00
/*@modifies fd @*/
{
int i ;
if ( buf ! = NULL & & buflen > 0 )
for ( i = fd - > ndigests - 1 ; i > = 0 ; i - - ) {
FDDIGEST_t fddig = fd - > digests + i ;
if ( fddig - > hashctx = = NULL )
continue ;
( void ) rpmDigestUpdate ( fddig - > hashctx , buf , buflen ) ;
}
}
/** \ingroup rpmio
*/
/*@unused@*/ static inline
void fdFiniDigest ( FD_t fd , pgpHashAlgo hashalgo ,
/*@null@*/ /*@out@*/ void * * datap ,
/*@null@*/ /*@out@*/ size_t * lenp ,
int asAscii )
/*@modifies fd, *datap, *lenp @*/
{
int imax = - 1 ;
int i ;
for ( i = fd - > ndigests - 1 ; i > = 0 ; i - - ) {
FDDIGEST_t fddig = fd - > digests + i ;
if ( fddig - > hashctx = = NULL )
continue ;
if ( i > imax ) imax = i ;
if ( fddig - > hashalgo ! = hashalgo )
continue ;
( void ) rpmDigestFinal ( fddig - > hashctx , datap , lenp , asAscii ) ;
fddig - > hashctx = NULL ;
break ;
}
if ( i < 0 ) {
if ( datap ) * datap = NULL ;
if ( lenp ) * lenp = 0 ;
2002-08-03 20:35:14 +04:00
}
fd - > ndigests = imax ;
if ( i < imax )
fd - > ndigests + + ; /* convert index to count */
2002-03-25 23:16:26 +03:00
}
/*@-shadow@*/
/** \ingroup rpmio
*/
/*@unused@*/ static inline
int fdFileno ( /*@null@*/ void * cookie )
/*@*/
{
FD_t fd ;
if ( cookie = = NULL ) return - 2 ;
fd = c2f ( cookie ) ;
return fd - > fps [ 0 ] . fdno ;
}
/*@=shadow@*/
/**
*/
int rpmioSlurp ( const char * fn ,
2002-08-03 20:35:14 +04:00
/*@out@*/ const unsigned char * * bp , /*@out@*/ ssize_t * blenp )
2002-03-25 23:16:26 +03:00
/*@globals fileSystem @*/
/*@modifies *bp, *blenp, fileSystem @*/ ;
2005-10-06 22:16:57 +04:00
typedef const char * ( * rpmBuiltinMacroLookup ) ( const char * , int ) ;
extern rpmBuiltinMacroLookup rpmSetBuiltinMacroLookup ( rpmBuiltinMacroLookup ) ;
2002-03-25 23:16:26 +03:00
# ifdef __cplusplus
}
# endif
# endif /* H_RPMIO_INTERNAL */