2001-12-06 14:10:51 +02:00
/* Copyright (C) 2000 MySQL AB
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
2000-07-31 21:29:14 +02:00
but WITHOUT ANY WARRANTY ; without even the implied warranty of
2001-12-06 14:10:51 +02:00
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 */
2000-07-31 21:29:14 +02:00
/* Return error-text for system error messages and nisam messages */
2004-05-25 22:00:14 +03:00
# define PERROR_VERSION "2.10"
2000-07-31 21:29:14 +02:00
2001-09-14 02:54:33 +03:00
# include <my_global.h>
2000-07-31 21:29:14 +02:00
# include <my_sys.h>
# include <m_string.h>
# include <errno.h>
2002-05-22 23:54:24 +03:00
# include <my_getopt.h>
2005-12-05 14:17:53 +01:00
# ifdef WITH_NDBCLUSTER_STORAGE_ENGINE
2005-04-26 19:04:36 -07:00
# include "../storage/ndb/src/ndbapi/ndberror.c"
2005-10-07 01:12:46 +02:00
# include "../storage/ndb/src/kernel/error/ndbd_exit_codes.c"
2004-05-05 14:19:47 +02:00
# endif
2000-07-31 21:29:14 +02:00
2002-05-22 23:54:24 +03:00
static my_bool verbose , print_all_codes ;
2000-07-31 21:29:14 +02:00
2005-12-05 14:17:53 +01:00
# ifdef WITH_NDBCLUSTER_STORAGE_ENGINE
2004-05-05 14:19:47 +02:00
static my_bool ndb_code ;
static char ndb_string [ 1024 ] ;
# endif
2002-05-22 23:54:24 +03:00
static struct my_option my_long_options [ ] =
2000-07-31 21:29:14 +02:00
{
2002-05-22 23:54:24 +03:00
{ " help " , ' ? ' , " Displays this help and exits. " , 0 , 0 , 0 , GET_NO_ARG ,
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2003-06-13 10:59:02 +02:00
{ " info " , ' I ' , " Synonym for --help. " , 0 , 0 , 0 , GET_NO_ARG ,
2002-05-22 23:54:24 +03:00
NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2005-12-05 14:17:53 +01:00
# ifdef WITH_NDBCLUSTER_STORAGE_ENGINE
2005-07-19 11:52:24 +02:00
{ " ndb " , 257 , " Ndbcluster storage engine specific error codes. " , ( gptr * ) & ndb_code ,
2004-08-18 10:10:50 +00:00
( gptr * ) & ndb_code , 0 , GET_BOOL , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
2004-05-05 14:19:47 +02:00
# endif
2002-05-22 23:54:24 +03:00
# ifdef HAVE_SYS_ERRLIST
{ " all " , ' a ' , " Print all the error messages and the number. " ,
( gptr * ) & print_all_codes , ( gptr * ) & print_all_codes , 0 , GET_BOOL , NO_ARG ,
0 , 0 , 0 , 0 , 0 , 0 } ,
# endif
2003-06-13 10:59:02 +02:00
{ " silent " , ' s ' , " Only print the error message. " , 0 , 0 , 0 , GET_NO_ARG , NO_ARG ,
2002-05-22 23:54:24 +03:00
0 , 0 , 0 , 0 , 0 , 0 } ,
{ " verbose " , ' v ' , " Print error code and message (default). " , ( gptr * ) & verbose ,
( gptr * ) & verbose , 0 , GET_BOOL , NO_ARG , 1 , 0 , 0 , 0 , 0 , 0 } ,
{ " version " , ' V ' , " Displays version information and exits. " ,
0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 } ,
{ 0 , 0 , 0 , 0 , 0 , 0 , GET_NO_ARG , NO_ARG , 0 , 0 , 0 , 0 , 0 , 0 }
2000-07-31 21:29:14 +02:00
} ;
2002-05-22 23:54:24 +03:00
2000-07-31 21:29:14 +02:00
typedef struct ha_errors {
int errcode ;
const char * msg ;
} HA_ERRORS ;
static HA_ERRORS ha_errlist [ ] =
{
{ 120 , " Didn't find key on read or update " } ,
{ 121 , " Duplicate key on write or update " } ,
2004-11-22 14:53:18 +01:00
{ 123 , " Someone has changed the row since it was read (while the table was locked to prevent it) " } ,
2000-07-31 21:29:14 +02:00
{ 124 , " Wrong index given to function " } ,
2003-01-28 17:42:08 +01:00
{ 126 , " Index file is crashed " } ,
2000-07-31 21:29:14 +02:00
{ 127 , " Record-file is crashed " } ,
2003-01-28 17:42:08 +01:00
{ 128 , " Out of memory " } ,
{ 130 , " Incorrect file format " } ,
2000-07-31 21:29:14 +02:00
{ 131 , " Command not supported by database " } ,
{ 132 , " Old database file " } ,
{ 133 , " No record read before update " } ,
{ 134 , " Record was already deleted (or record file crashed) " } ,
{ 135 , " No more room in record file " } ,
{ 136 , " No more room in index file " } ,
{ 137 , " No more records (read after end of file) " } ,
{ 138 , " Unsupported extension used for table " } ,
2004-04-27 23:03:25 +02:00
{ 139 , " Too big row " } ,
2000-07-31 21:29:14 +02:00
{ 140 , " Wrong create options " } ,
2000-10-14 03:16:35 +03:00
{ 141 , " Duplicate unique key or constraint on write or update " } ,
2000-09-14 02:39:07 +03:00
{ 142 , " Unknown character set used " } ,
2003-01-28 17:42:08 +01:00
{ 143 , " Conflicting table definitions in sub-tables of MERGE table " } ,
2000-09-26 00:33:25 +03:00
{ 144 , " Table is crashed and last repair failed " } ,
2000-11-21 03:43:34 +02:00
{ 145 , " Table was marked as crashed and should be repaired " } ,
2001-09-17 23:43:53 +03:00
{ 146 , " Lock timed out; Retry transaction " } ,
{ 147 , " Lock table is full; Restart program with a larger locktable " } ,
{ 148 , " Updates are not allowed under a read only transactions " } ,
{ 149 , " Lock deadlock; Retry transaction " } ,
2002-12-14 22:38:33 +02:00
{ 150 , " Foreign key constraint is incorrectly formed " } ,
2003-01-28 17:42:08 +01:00
{ 151 , " Cannot add a child row " } ,
{ 152 , " Cannot delete a parent row " } ,
2001-09-18 04:58:15 +03:00
{ - 30999 , " DB_INCOMPLETE: Sync didn't finish " } ,
{ - 30998 , " DB_KEYEMPTY: Key/data deleted or never created " } ,
{ - 30997 , " DB_KEYEXIST: The key/data pair already exists " } ,
{ - 30996 , " DB_LOCK_DEADLOCK: Deadlock " } ,
{ - 30995 , " DB_LOCK_NOTGRANTED: Lock unavailable " } ,
{ - 30994 , " DB_NOSERVER: Server panic return " } ,
{ - 30993 , " DB_NOSERVER_HOME: Bad home sent to server " } ,
{ - 30992 , " DB_NOSERVER_ID: Bad ID sent to server " } ,
{ - 30991 , " DB_NOTFOUND: Key/data pair not found (EOF) " } ,
{ - 30990 , " DB_OLD_VERSION: Out-of-date version " } ,
{ - 30989 , " DB_RUNRECOVERY: Panic return " } ,
{ - 30988 , " DB_VERIFY_BAD: Verify failed; bad format " } ,
2000-07-31 21:29:14 +02:00
{ 0 , NullS } ,
} ;
2005-02-08 19:49:40 +02:00
# include <help_start.h>
2000-07-31 21:29:14 +02:00
static void print_version ( void )
{
printf ( " %s Ver %s, for %s (%s) \n " , my_progname , PERROR_VERSION ,
SYSTEM_TYPE , MACHINE_TYPE ) ;
}
2005-02-08 19:49:40 +02:00
2000-07-31 21:29:14 +02:00
static void usage ( void )
{
print_version ( ) ;
puts ( " This software comes with ABSOLUTELY NO WARRANTY. This is free software, \n and you are welcome to modify and redistribute it under the GPL license \n " ) ;
2003-07-12 11:45:43 -05:00
printf ( " Print a description for a system error code or an error code from \n a MyISAM/ISAM/BDB table handler. \n " ) ;
2001-09-18 04:58:15 +03:00
printf ( " If you want to get the error for a negative error code, you should use \n -- before the first error code to tell perror that there was no more options. \n \n " ) ;
2000-10-11 00:48:03 +03:00
printf ( " Usage: %s [OPTIONS] [ERRORCODE [ERRORCODE...]] \n " , my_progname ) ;
2002-05-22 23:54:24 +03:00
my_print_help ( my_long_options ) ;
my_print_variables ( my_long_options ) ;
}
2005-02-08 19:49:40 +02:00
# include <help_end.h>
2002-05-22 23:54:24 +03:00
static my_bool
get_one_option ( int optid , const struct my_option * opt __attribute__ ( ( unused ) ) ,
char * argument __attribute__ ( ( unused ) ) )
{
switch ( optid ) {
case ' s ' :
verbose = 0 ;
break ;
case ' V ' :
print_version ( ) ;
exit ( 0 ) ;
break ;
case ' I ' :
case ' ? ' :
usage ( ) ;
exit ( 0 ) ;
break ;
}
return 0 ;
2001-12-06 14:10:51 +02:00
}
2000-07-31 21:29:14 +02:00
static int get_options ( int * argc , char * * * argv )
{
2002-05-22 23:54:24 +03:00
int ho_error ;
2000-07-31 21:29:14 +02:00
2004-08-31 21:27:58 +05:00
if ( ( ho_error = handle_options ( argc , argv , my_long_options , get_one_option ) ) )
2002-05-29 15:07:30 +03:00
exit ( ho_error ) ;
2000-07-31 21:29:14 +02:00
if ( ! * argc & & ! print_all_codes )
{
usage ( ) ;
return 1 ;
}
return 0 ;
} /* get_options */
static const char * get_ha_error_msg ( int code )
{
HA_ERRORS * ha_err_ptr ;
for ( ha_err_ptr = ha_errlist ; ha_err_ptr - > errcode ; ha_err_ptr + + )
if ( ha_err_ptr - > errcode = = code )
return ha_err_ptr - > msg ;
return NullS ;
}
int main ( int argc , char * argv [ ] )
{
int error , code , found ;
const char * msg ;
2005-01-31 23:02:32 -06:00
char * unknown_error = 0 ;
2000-07-31 21:29:14 +02:00
MY_INIT ( argv [ 0 ] ) ;
if ( get_options ( & argc , & argv ) )
exit ( 1 ) ;
error = 0 ;
# ifdef HAVE_SYS_ERRLIST
if ( print_all_codes )
{
HA_ERRORS * ha_err_ptr ;
for ( code = 1 ; code < sys_nerr ; code + + )
{
if ( sys_errlist [ code ] [ 0 ] )
2004-06-03 11:52:54 -05:00
{ /* Skip if no error-text */
2000-07-31 21:29:14 +02:00
printf ( " %3d = %s \n " , code , sys_errlist [ code ] ) ;
}
}
for ( ha_err_ptr = ha_errlist ; ha_err_ptr - > errcode ; ha_err_ptr + + )
printf ( " %3d = %s \n " , ha_err_ptr - > errcode , ha_err_ptr - > msg ) ;
}
else
# endif
{
2004-05-25 22:00:14 +03:00
/*
On some system , like NETWARE , strerror ( unknown_error ) returns a
string ' Unknown Error ' . To avoid printing it we try to find the
error string by asking for an impossible big error message .
*/
2005-02-02 20:28:01 +02:00
msg = strerror ( 10000 ) ;
2005-01-31 23:02:32 -06:00
2005-02-02 20:28:01 +02:00
/*
Allocate a buffer for unknown_error since strerror always returns
the same pointer on some platforms such as Windows
*/
unknown_error = malloc ( strlen ( msg ) + 1 ) ;
strmov ( unknown_error , msg ) ;
2004-05-25 22:00:14 +03:00
2000-07-31 21:29:14 +02:00
for ( ; argc - - > 0 ; argv + + )
{
2004-05-25 22:54:00 +03:00
2000-07-31 21:29:14 +02:00
found = 0 ;
code = atoi ( * argv ) ;
2005-12-05 14:17:53 +01:00
# ifdef WITH_NDBCLUSTER_STORAGE_ENGINE
2004-05-05 14:19:47 +02:00
if ( ndb_code )
{
2005-10-03 20:04:44 +02:00
if ( ( ndb_error_string ( code , ndb_string , sizeof ( ndb_string ) ) < 0 ) & &
( ndbd_exit_string ( code , ndb_string , sizeof ( ndb_string ) ) < 0 ) )
{
2006-02-08 17:17:27 +01:00
msg = 0 ;
2005-10-03 20:04:44 +02:00
}
2004-05-05 14:19:47 +02:00
else
msg = ndb_string ;
2006-02-08 17:17:27 +01:00
if ( msg )
{
if ( verbose )
printf ( " NDB error code %3d: %s \n " , code , msg ) ;
else
puts ( msg ) ;
}
else
{
fprintf ( stderr , " Illegal ndb error code: %d \n " , code ) ;
error = 1 ;
}
found = 1 ;
msg = 0 ;
2004-05-05 14:19:47 +02:00
}
else
# endif
msg = strerror ( code ) ;
2004-05-25 22:00:14 +03:00
/*
2005-02-16 17:37:56 -08:00
We don ' t print the OS error message if it is the same as the
unknown_error message we retrieved above , or it starts with
' Unknown Error ' ( without regard to case ) .
2004-05-25 22:00:14 +03:00
*/
2005-02-16 17:37:56 -08:00
if ( msg & &
2005-02-25 16:53:22 +02:00
my_strnncoll ( & my_charset_latin1 , ( const uchar * ) msg , 13 ,
( const uchar * ) " Unknown Error " , 13 ) & &
2004-05-25 22:00:14 +03:00
( ! unknown_error | | strcmp ( msg , unknown_error ) ) )
2000-07-31 21:29:14 +02:00
{
found = 1 ;
if ( verbose )
2005-02-16 17:37:56 -08:00
printf ( " OS error code %3d: %s \n " , code , msg ) ;
2000-07-31 21:29:14 +02:00
else
puts ( msg ) ;
}
if ( ! ( msg = get_ha_error_msg ( code ) ) )
{
if ( ! found )
{
fprintf ( stderr , " Illegal error code: %d \n " , code ) ;
error = 1 ;
}
}
else
{
if ( verbose )
2005-02-16 17:37:56 -08:00
printf ( " MySQL error code %3d: %s \n " , code , msg ) ;
2000-07-31 21:29:14 +02:00
else
puts ( msg ) ;
}
}
}
2005-01-31 23:02:32 -06:00
/* if we allocated a buffer for unknown_error, free it now */
if ( unknown_error )
2005-02-02 20:28:01 +02:00
free ( unknown_error ) ;
2005-01-31 23:02:32 -06:00
2000-07-31 21:29:14 +02:00
exit ( error ) ;
return error ;
}