2011-01-16 10:26:25 +03:00
/*
* checkFiles . c - check packaged file list .
* Written by Alexey Tourbin < at @ altlinux . org > .
* License : GPLv2 + .
*/
2011-01-15 10:58:41 +03:00
# include "system.h"
# include "rpmio_internal.h"
# include "rpmbuild.h"
# include "psm.h"
# include "checkFiles.h"
2015-01-28 16:52:29 +00:00
void fiIntersect ( const TFI_t fi1 , const TFI_t fi2 ,
void ( * cb ) ( const TFI_t fi1 , const TFI_t fi2 ,
const char * f , int i1 , int i2 , void * data ) ,
void * data )
2011-01-15 11:49:07 +03:00
{
2015-01-28 16:52:29 +00:00
if ( ! fi1 | | ! fi2 ) return ;
2011-01-15 11:49:07 +03:00
int i1 = 0 , i2 = 0 ;
while ( i1 < fi1 - > fc & & i2 < fi2 - > fc ) {
char f1 [ PATH_MAX ] , f2 [ PATH_MAX ] ;
strcpy ( f1 , fi1 - > dnl [ fi1 - > dil [ i1 ] ] + fi1 - > astriplen ) ;
strcpy ( f2 , fi2 - > dnl [ fi2 - > dil [ i2 ] ] + fi2 - > astriplen ) ;
strcat ( f1 , fi1 - > bnl [ i1 ] ) ;
strcat ( f2 , fi2 - > bnl [ i2 ] ) ;
int cmp = strcmp ( f1 , f2 ) ;
if ( cmp < 0 ) {
i1 + + ;
continue ;
}
if ( cmp > 0 ) {
i2 + + ;
continue ;
}
2015-01-28 16:52:29 +00:00
cb ( fi1 , fi2 , f1 , i1 , i2 , data ) ;
2011-01-15 11:49:07 +03:00
i1 + + ;
i2 + + ;
}
}
static
2015-01-28 16:52:29 +00:00
void fiIntersect_cb ( const TFI_t fi1 , const TFI_t fi2 , const char * f ,
int i1 , int i2 , void * data )
2011-01-15 11:49:07 +03:00
{
if ( S_ISDIR ( fi1 - > fmodes [ i1 ] ) & & S_ISDIR ( fi2 - > fmodes [ i2 ] ) )
2015-01-28 16:52:29 +00:00
return ;
2011-02-05 23:41:05 +03:00
const char src [ ] = " /usr/src/debug/ " ;
if ( strncmp ( f , src , sizeof ( src ) - 1 ) = = 0 )
2015-01-28 16:52:29 +00:00
return ;
int * once = data ;
if ( ! * once ) {
rpmlog ( RPMLOG_WARNING ,
" File(s) packaged into both %s-%s-%s and %s-%s-%s: \n " ,
fi1 - > name , fi1 - > version , fi1 - > release ,
fi2 - > name , fi2 - > version , fi2 - > release ) ;
* once = 1 ;
}
2011-01-15 11:49:07 +03:00
rpmlog ( RPMLOG_INFO , " %s \n " , f ) ;
2015-01-28 16:52:29 +00:00
}
static
void checkPkgIntersect ( Package pkg1 , Package pkg2 )
{
const TFI_t fi1 = pkg1 - > cpioList ;
const TFI_t fi2 = pkg2 - > cpioList ;
if ( ! fi1 | | ! fi2 ) return ;
int once = 0 ;
fiIntersect ( fi1 , fi2 , fiIntersect_cb , ( void * ) & once ) ;
2011-01-15 11:49:07 +03:00
}
static
void checkIntersect ( Spec spec )
{
Package pkg1 , pkg2 ;
for ( pkg1 = spec - > packages ; pkg1 ; pkg1 = pkg1 - > next )
for ( pkg2 = pkg1 - > next ; pkg2 ; pkg2 = pkg2 - > next )
checkPkgIntersect ( pkg1 , pkg2 ) ;
}
2011-01-15 10:58:41 +03:00
static
int fiSearch ( TFI_t fi , const char * path )
{
if ( fi = = NULL | | fi - > fc = = 0 )
return - 1 ;
int l = 0 ;
int u = fi - > fc ;
while ( l < u ) {
int i = ( l + u ) / 2 ;
const char * d = fi - > dnl [ fi - > dil [ i ] ] ;
int dlen = strlen ( d ) ;
int cmp = strncmp ( path , d , dlen ) ;
if ( cmp = = 0 ) {
const char * b = fi - > bnl [ i ] ;
cmp = strcmp ( path + dlen , b ) ;
}
if ( cmp < 0 )
u = i ;
else if ( cmp > 0 )
l = i + 1 ;
else
return i ;
}
return - 1 ;
}
# include <fts.h>
static
2015-01-28 16:52:29 +00:00
int avcmp ( const void * sptr1 , const void * sptr2 )
2011-01-15 10:58:41 +03:00
{
2011-01-21 20:45:22 +03:00
const char * s1 = * ( const char * * ) sptr1 ;
const char * s2 = * ( const char * * ) sptr2 ;
return strcmp ( s1 , s2 ) ;
2015-01-28 16:52:29 +00:00
}
static
int check ( const char * path , const Spec spec , unsigned int * uc , char * * * uv )
{
2011-01-15 10:58:41 +03:00
Package pkg ;
for ( pkg = spec - > packages ; pkg ; pkg = pkg - > next )
if ( fiSearch ( pkg - > cpioList , path ) > = 0 )
2015-01-28 16:52:29 +00:00
return 0 ;
2011-01-21 20:45:22 +03:00
if ( spec - > exclude ) {
const char * key = path + strlen ( spec - > buildRootURL ) ;
if ( bsearch ( & key , spec - > exclude , spec - > excludeCount ,
sizeof ( char * ) , avcmp ) )
2015-01-28 16:52:29 +00:00
return 0 ;
2011-01-21 20:45:22 +03:00
}
2015-01-28 16:52:29 +00:00
AUTO_REALLOC ( * uv , * uc , 8 ) ;
( * uv ) [ ( * uc ) + + ] = xstrdup ( path + strlen ( spec - > buildRootURL ) ) ;
return 1 ;
}
static
int checkUnpackaged ( Spec spec )
{
int rc = 0 ;
unsigned int uc = 0 ;
char * * uv = NULL ;
if ( spec - > exclude )
qsort ( spec - > exclude , spec - > excludeCount , sizeof ( char * ) , avcmp ) ;
2011-01-15 10:58:41 +03:00
char * paths [ ] = { ( char * ) spec - > buildRootURL , 0 } ;
int options = FTS_COMFOLLOW | FTS_PHYSICAL ;
FTS * ftsp = fts_open ( paths , options , NULL ) ;
FTSENT * fts ;
while ( ( fts = fts_read ( ftsp ) ) ! = NULL ) {
2011-01-23 02:18:28 +03:00
// buildroot may not exist
if ( fts - > fts_level = = 0 & & fts - > fts_info = = FTS_NS ) {
Package pkg ;
for ( pkg = spec - > packages ; pkg ; pkg = pkg - > next ) {
TFI_t fi = pkg - > cpioList ;
if ( fi & & fi - > fc > 0 )
break ;
}
if ( pkg = = NULL )
continue ;
}
2011-01-15 10:58:41 +03:00
// skip dotfiles in buildroot
if ( fts - > fts_level = = 1 & & * fts - > fts_name = = ' . ' ) {
if ( fts - > fts_info = = FTS_D )
fts_set ( ftsp , fts , FTS_SKIP ) ;
continue ;
}
2011-01-30 03:54:54 +03:00
// skip debuginfo
if ( fts - > fts_level = = 3 & & fts - > fts_info = = FTS_D ) {
const char * dir = fts - > fts_path + strlen ( spec - > buildRootURL ) ;
if ( strcmp ( dir , " /usr/lib/debug " ) = = 0 | |
strcmp ( dir , " /usr/src/debug " ) = = 0 )
{
fts_set ( ftsp , fts , FTS_SKIP ) ;
continue ;
}
}
2011-01-15 10:58:41 +03:00
switch ( fts - > fts_info ) {
// do not check for unpackaged directories
case FTS_D :
case FTS_DP :
break ;
case FTS_F :
case FTS_SL :
case FTS_SLNONE :
case FTS_DEFAULT :
2015-01-28 16:52:29 +00:00
rc | = check ( fts - > fts_path , spec , & uc , & uv ) ;
2011-01-15 10:58:41 +03:00
break ;
default :
rpmlog ( RPMLOG_WARNING , " %s: fts error \n " , fts - > fts_path ) ;
rc | = 2 ;
break ;
}
}
if ( uv ) {
rpmlog ( RPMLOG_WARNING , " Installed (but unpackaged) file(s) found: \n " ) ;
2011-01-21 20:45:22 +03:00
qsort ( uv , uc , sizeof ( * uv ) , avcmp ) ;
2015-01-28 16:52:29 +00:00
unsigned int i ;
2011-01-15 10:58:41 +03:00
for ( i = 0 ; i < uc ; i + + )
rpmlog ( RPMLOG_INFO , " %s \n " , uv [ i ] ) ;
free ( uv ) ;
}
return rc ;
}
int checkFiles ( Spec spec )
{
2011-01-15 11:49:07 +03:00
checkIntersect ( spec ) ;
2011-01-15 10:58:41 +03:00
int rc = checkUnpackaged ( spec ) ;
if ( rc & & rpmExpandNumeric ( " %{?_unpackaged_files_terminate_build} " ) ) {
rpmlog ( RPMLOG_ERR , " File list check failed, terminating build \n " ) ;
return rc ;
}
return 0 ;
}
// ex: set ts=8 sts=4 sw=4 noet: