diff options
Diffstat (limited to 'src')
| -rw-r--r-- | src/Makefile.am | 39 | ||||
| -rw-r--r-- | src/config.h | 94 | ||||
| -rw-r--r-- | src/cvtendian.h | 48 | ||||
| -rw-r--r-- | src/gettext.h | 271 | ||||
| -rw-r--r-- | src/libr-backends.h | 22 | ||||
| -rw-r--r-- | src/libr-bfd.c | 533 | ||||
| -rw-r--r-- | src/libr-bfd.h | 40 | ||||
| -rw-r--r-- | src/libr-elf.c | 412 | ||||
| -rw-r--r-- | src/libr-elf.h | 24 | ||||
| -rw-r--r-- | src/libr-gtk.c | 443 | ||||
| -rw-r--r-- | src/libr-gtk.h | 55 | ||||
| -rw-r--r-- | src/libr-i18n.c | 84 | ||||
| -rw-r--r-- | src/libr-i18n.h | 14 | ||||
| -rw-r--r-- | src/libr-icons.c | 643 | ||||
| -rw-r--r-- | src/libr-icons.h | 201 | ||||
| -rw-r--r-- | src/libr-internal.h | 34 | ||||
| -rw-r--r-- | src/libr-link.h | 26 | ||||
| -rw-r--r-- | src/libr-ro.c | 351 | ||||
| -rw-r--r-- | src/libr-ro.h | 62 | ||||
| -rw-r--r-- | src/libr.c | 489 | ||||
| -rw-r--r-- | src/libr.h | 416 | ||||
| -rw-r--r-- | src/onecanvas.c | 446 | ||||
| -rw-r--r-- | src/onecanvas.h | 6 | ||||
| -rw-r--r-- | src/tempfiles.c | 317 | ||||
| -rw-r--r-- | src/tempfiles.h | 13 | 
25 files changed, 5083 insertions, 0 deletions
| diff --git a/src/Makefile.am b/src/Makefile.am new file mode 100644 index 0000000..5fbf00b --- /dev/null +++ b/src/Makefile.am @@ -0,0 +1,39 @@ +libr_la_includedir = $(includedir)/libr +LIBTOOL_DEPS = @LIBTOOL_DEPS@ + +INCLUDES = \ +	-D__LIBR_BACKEND_@BACKEND_NAME@__ \ +	-D__LIBR_BUILD__ \ +	@LIBGLADE_CFLAGS@ \ +	@BACKEND_CFLAGS@ \ +	@EXTRA_CFLAGS@ + +lib_LTLIBRARIES = \ +	libr.la + +libr_la_SOURCES = \ +	libr-@LIBR_BACKEND@.c \ +	tempfiles.c \ +	onecanvas.c \ +	libr-icons.c \ +	libr-i18n.c \ +	libr-gtk.c \ +	libr.c + +libr_la_include_HEADERS = \ +	gettext.h \ +	libr-icons.h \ +	libr-i18n.h \ +	libr-gtk.h \ +	libr.h + +libr_la_LIBADD = \ +	@BACKEND_LIBS@ \ +	@EXTRA_LIBS@ + +# If not in a fakeroot environment then run ldconfig +install: install-am +	@if [ ! -n "${FAKEROOTKEY}" ]; then \ +		echo "Regenerating system dependencies..."; \ +		ldconfig; \ +	fi diff --git a/src/config.h b/src/config.h new file mode 100644 index 0000000..a51743e --- /dev/null +++ b/src/config.h @@ -0,0 +1,94 @@ +/* config.h.  Generated from config.h.in by configure.  */ +/* config.h.in.  Generated from configure.ac by autoheader.  */ + +/* Define to 1 if translation of program messages to the user's native +   language is requested. */ +#define ENABLE_NLS 1 + +/* Define to 1 if you have the MacOS X function CFLocaleCopyCurrent in the +   CoreFoundation framework. */ +/* #undef HAVE_CFLOCALECOPYCURRENT */ + +/* Define to 1 if you have the MacOS X function CFPreferencesCopyAppValue in +   the CoreFoundation framework. */ +/* #undef HAVE_CFPREFERENCESCOPYAPPVALUE */ + +/* Define if the GNU dcgettext() function is already present or preinstalled. +   */ +#define HAVE_DCGETTEXT 1 + +/* Define to 1 if you have the <dlfcn.h> header file. */ +#define HAVE_DLFCN_H 1 + +/* Define if the GNU gettext() function is already present or preinstalled. */ +#define HAVE_GETTEXT 1 + +/* Define if you have the iconv() function and it works. */ +/* #undef HAVE_ICONV */ + +/* Define to 1 if you have the <inttypes.h> header file. */ +#define HAVE_INTTYPES_H 1 + +/* Define to 1 if you have the <math.h> header file. */ +#define HAVE_MATH_H 1 + +/* Define to 1 if you have the <memory.h> header file. */ +#define HAVE_MEMORY_H 1 + +/* Define to 1 if you have the <pthread.h> header file. */ +#define HAVE_PTHREAD_H 1 + +/* Define to 1 if you have the <stdint.h> header file. */ +#define HAVE_STDINT_H 1 + +/* Define to 1 if you have the <stdlib.h> header file. */ +#define HAVE_STDLIB_H 1 + +/* Define to 1 if you have the <strings.h> header file. */ +#define HAVE_STRINGS_H 1 + +/* Define to 1 if you have the <string.h> header file. */ +#define HAVE_STRING_H 1 + +/* Define to 1 if you have the <sys/stat.h> header file. */ +#define HAVE_SYS_STAT_H 1 + +/* Define to 1 if you have the <sys/types.h> header file. */ +#define HAVE_SYS_TYPES_H 1 + +/* Define to 1 if you have the <unistd.h> header file. */ +#define HAVE_UNISTD_H 1 + +/* Define to 1 if you have the <zlib.h> header file. */ +#define HAVE_ZLIB_H 1 + +/* Define to the sub-directory in which libtool stores uninstalled libraries. +   */ +#define LT_OBJDIR ".libs/" + +/* Name of package */ +#define PACKAGE "libr" + +/* Define to the address where bug reports for this package should be sent. */ +#define PACKAGE_BUGREPORT "" + +/* Define to the full name of this package. */ +#define PACKAGE_NAME "" + +/* Define to the full name and version of this package. */ +#define PACKAGE_STRING "" + +/* Define to the one symbol short name of this package. */ +#define PACKAGE_TARNAME "" + +/* Define to the home page for this package. */ +#define PACKAGE_URL "" + +/* Define to the version of this package. */ +#define PACKAGE_VERSION "" + +/* Define to 1 if you have the ANSI C header files. */ +#define STDC_HEADERS 1 + +/* Version number of package */ +#define VERSION "0.4.0" diff --git a/src/cvtendian.h b/src/cvtendian.h new file mode 100644 index 0000000..d69158b --- /dev/null +++ b/src/cvtendian.h @@ -0,0 +1,48 @@ +#ifndef __CVTENDIAN_H +#define __CVTENDIAN_H + +/* Support for swapping bytes (endian conversion) */ +#include <byteswap.h> + +/* For obtaining the host endian type */ +#include <endian.h> +#if (__BYTE_ORDER == __LITTLE_ENDIAN) +#	define HOST_ENDIAN  ELFDATA2LSB +#elif (__BYTE_ORDER == __BIG_ENDIAN) +#	define HOST_ENDIAN  ELFDATA2MSB +#else +#	error "Failed to detect host endian type" +#endif + +/* + * Convert the endian of a parameter + */ +static int ConvertEndian(void *ptr, int bytes) +{ +	switch(bytes) +	{ +		case 2: +		{ +			uint16_t *value = (uint16_t *) ptr; +			 +			*value = bswap_16(*value); +		}	return 1; +		case 4: +		{ +			uint32_t *value = (uint32_t *) ptr; +			 +			*value = bswap_32(*value); +		}	return 1; +		case 8: +		{ +			uint64_t *value = (uint64_t *) ptr; +			 +			*value = bswap_64(*value); +		}	return 1; +		default: +			break; +	} +	return 0; +} + +#endif /* __CVTENDIAN_H */ diff --git a/src/gettext.h b/src/gettext.h new file mode 100644 index 0000000..209921e --- /dev/null +++ b/src/gettext.h @@ -0,0 +1,271 @@ +/* Convenience header for conditional use of GNU <libintl.h>. +   Copyright (C) 1995-1998, 2000-2002, 2004-2006 Free Software Foundation, Inc. + +   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, or (at your option) +   any later version. + +   This program is distributed in the hope that it will be useful, +   but WITHOUT ANY WARRANTY; without even the implied warranty of +   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU +   Library 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, +   USA.  */ + +#ifndef _LIBGETTEXT_H +#define _LIBGETTEXT_H 1 + +/* NLS can be disabled through the configure --disable-nls option.  */ +#if ENABLE_NLS + +/* Get declarations of GNU message catalog functions.  */ +# include <libintl.h> + +/* You can set the DEFAULT_TEXT_DOMAIN macro to specify the domain used by +   the gettext() and ngettext() macros.  This is an alternative to calling +   textdomain(), and is useful for libraries.  */ +# ifdef DEFAULT_TEXT_DOMAIN +#  undef gettext +#  define gettext(Msgid) \ +     dgettext (DEFAULT_TEXT_DOMAIN, Msgid) +#  undef ngettext +#  define ngettext(Msgid1, Msgid2, N) \ +     dngettext (DEFAULT_TEXT_DOMAIN, Msgid1, Msgid2, N) +# endif + +#else + +/* Solaris /usr/include/locale.h includes /usr/include/libintl.h, which +   chokes if dcgettext is defined as a macro.  So include it now, to make +   later inclusions of <locale.h> a NOP.  We don't include <libintl.h> +   as well because people using "gettext.h" will not include <libintl.h>, +   and also including <libintl.h> would fail on SunOS 4, whereas <locale.h> +   is OK.  */ +#if defined(__sun) +# include <locale.h> +#endif + +/* Many header files from the libstdc++ coming with g++ 3.3 or newer include +   <libintl.h>, which chokes if dcgettext is defined as a macro.  So include +   it now, to make later inclusions of <libintl.h> a NOP.  */ +#if defined(__cplusplus) && defined(__GNUG__) && (__GNUC__ >= 3) +# include <cstdlib> +# if (__GLIBC__ >= 2) || _GLIBCXX_HAVE_LIBINTL_H +#  include <libintl.h> +# endif +#endif + +/* Disabled NLS. +   The casts to 'const char *' serve the purpose of producing warnings +   for invalid uses of the value returned from these functions. +   On pre-ANSI systems without 'const', the config.h file is supposed to +   contain "#define const".  */ +# define gettext(Msgid) ((const char *) (Msgid)) +# define dgettext(Domainname, Msgid) ((void) (Domainname), gettext (Msgid)) +# define dcgettext(Domainname, Msgid, Category) \ +    ((void) (Category), dgettext (Domainname, Msgid)) +# define ngettext(Msgid1, Msgid2, N) \ +    ((N) == 1 \ +     ? ((void) (Msgid2), (const char *) (Msgid1)) \ +     : ((void) (Msgid1), (const char *) (Msgid2))) +# define dngettext(Domainname, Msgid1, Msgid2, N) \ +    ((void) (Domainname), ngettext (Msgid1, Msgid2, N)) +# define dcngettext(Domainname, Msgid1, Msgid2, N, Category) \ +    ((void) (Category), dngettext(Domainname, Msgid1, Msgid2, N)) +# define textdomain(Domainname) ((const char *) (Domainname)) +# define bindtextdomain(Domainname, Dirname) \ +    ((void) (Domainname), (const char *) (Dirname)) +# define bind_textdomain_codeset(Domainname, Codeset) \ +    ((void) (Domainname), (const char *) (Codeset)) + +#endif + +/* A pseudo function call that serves as a marker for the automated +   extraction of messages, but does not call gettext().  The run-time +   translation is done at a different place in the code. +   The argument, String, should be a literal string.  Concatenated strings +   and other string expressions won't work. +   The macro's expansion is not parenthesized, so that it is suitable as +   initializer for static 'char[]' or 'const char[]' variables.  */ +#define gettext_noop(String) String + +/* The separator between msgctxt and msgid in a .mo file.  */ +#define GETTEXT_CONTEXT_GLUE "\004" + +/* Pseudo function calls, taking a MSGCTXT and a MSGID instead of just a +   MSGID.  MSGCTXT and MSGID must be string literals.  MSGCTXT should be +   short and rarely need to change. +   The letter 'p' stands for 'particular' or 'special'.  */ +#ifdef DEFAULT_TEXT_DOMAIN +# define pgettext(Msgctxt, Msgid) \ +   pgettext_aux (DEFAULT_TEXT_DOMAIN, Msgctxt GETTEXT_CONTEXT_GLUE Msgid, Msgid, LC_MESSAGES) +#else +# define pgettext(Msgctxt, Msgid) \ +   pgettext_aux (NULL, Msgctxt GETTEXT_CONTEXT_GLUE Msgid, Msgid, LC_MESSAGES) +#endif +#define dpgettext(Domainname, Msgctxt, Msgid) \ +  pgettext_aux (Domainname, Msgctxt GETTEXT_CONTEXT_GLUE Msgid, Msgid, LC_MESSAGES) +#define dcpgettext(Domainname, Msgctxt, Msgid, Category) \ +  pgettext_aux (Domainname, Msgctxt GETTEXT_CONTEXT_GLUE Msgid, Msgid, Category) +#ifdef DEFAULT_TEXT_DOMAIN +# define npgettext(Msgctxt, Msgid, MsgidPlural, N) \ +   npgettext_aux (DEFAULT_TEXT_DOMAIN, Msgctxt GETTEXT_CONTEXT_GLUE Msgid, Msgid, MsgidPlural, N, LC_MESSAGES) +#else +# define npgettext(Msgctxt, Msgid, MsgidPlural, N) \ +   npgettext_aux (NULL, Msgctxt GETTEXT_CONTEXT_GLUE Msgid, Msgid, MsgidPlural, N, LC_MESSAGES) +#endif +#define dnpgettext(Domainname, Msgctxt, Msgid, MsgidPlural, N) \ +  npgettext_aux (Domainname, Msgctxt GETTEXT_CONTEXT_GLUE Msgid, Msgid, MsgidPlural, N, LC_MESSAGES) +#define dcnpgettext(Domainname, Msgctxt, Msgid, MsgidPlural, N, Category) \ +  npgettext_aux (Domainname, Msgctxt GETTEXT_CONTEXT_GLUE Msgid, Msgid, MsgidPlural, N, Category) + +#ifdef __GNUC__ +__inline +#else +#ifdef __cplusplus +inline +#endif +#endif +static const char * +pgettext_aux (const char *domain, +	      const char *msg_ctxt_id, const char *msgid, +	      int category) +{ +  const char *translation = dcgettext (domain, msg_ctxt_id, category); +  if (translation == msg_ctxt_id) +    return msgid; +  else +    return translation; +} + +#ifdef __GNUC__ +__inline +#else +#ifdef __cplusplus +inline +#endif +#endif +static const char * +npgettext_aux (const char *domain, +	       const char *msg_ctxt_id, const char *msgid, +	       const char *msgid_plural, unsigned long int n, +	       int category) +{ +  const char *translation = +    dcngettext (domain, msg_ctxt_id, msgid_plural, n, category); +  if (translation == msg_ctxt_id || translation == msgid_plural) +    return (n == 1 ? msgid : msgid_plural); +  else +    return translation; +} + +/* The same thing extended for non-constant arguments.  Here MSGCTXT and MSGID +   can be arbitrary expressions.  But for string literals these macros are +   less efficient than those above.  */ + +#include <string.h> + +#define _LIBGETTEXT_HAVE_VARIABLE_SIZE_ARRAYS \ +  (((__GNUC__ >= 3 || __GNUG__ >= 2) && !__STRICT_ANSI__) \ +   /* || __STDC_VERSION__ >= 199901L */ ) + +#if !_LIBGETTEXT_HAVE_VARIABLE_SIZE_ARRAYS +#include <stdlib.h> +#endif + +#define pgettext_expr(Msgctxt, Msgid) \ +  dcpgettext_expr (NULL, Msgctxt, Msgid, LC_MESSAGES) +#define dpgettext_expr(Domainname, Msgctxt, Msgid) \ +  dcpgettext_expr (Domainname, Msgctxt, Msgid, LC_MESSAGES) + +#ifdef __GNUC__ +__inline +#else +#ifdef __cplusplus +inline +#endif +#endif +static const char * +dcpgettext_expr (const char *domain, +		 const char *msgctxt, const char *msgid, +		 int category) +{ +  size_t msgctxt_len = strlen (msgctxt) + 1; +  size_t msgid_len = strlen (msgid) + 1; +  const char *translation; +#if _LIBGETTEXT_HAVE_VARIABLE_SIZE_ARRAYS +  char msg_ctxt_id[msgctxt_len + msgid_len]; +#else +  char buf[1024]; +  char *msg_ctxt_id = +    (msgctxt_len + msgid_len <= sizeof (buf) +     ? buf +     : (char *) malloc (msgctxt_len + msgid_len)); +  if (msg_ctxt_id != NULL) +#endif +    { +      memcpy (msg_ctxt_id, msgctxt, msgctxt_len - 1); +      msg_ctxt_id[msgctxt_len - 1] = '\004'; +      memcpy (msg_ctxt_id + msgctxt_len, msgid, msgid_len); +      translation = dcgettext (domain, msg_ctxt_id, category); +#if !_LIBGETTEXT_HAVE_VARIABLE_SIZE_ARRAYS +      if (msg_ctxt_id != buf) +	free (msg_ctxt_id); +#endif +      if (translation != msg_ctxt_id) +	return translation; +    } +  return msgid; +} + +#define npgettext_expr(Msgctxt, Msgid, MsgidPlural, N) \ +  dcnpgettext_expr (NULL, Msgctxt, Msgid, MsgidPlural, N, LC_MESSAGES) +#define dnpgettext_expr(Domainname, Msgctxt, Msgid, MsgidPlural, N) \ +  dcnpgettext_expr (Domainname, Msgctxt, Msgid, MsgidPlural, N, LC_MESSAGES) + +#ifdef __GNUC__ +__inline +#else +#ifdef __cplusplus +inline +#endif +#endif +static const char * +dcnpgettext_expr (const char *domain, +		  const char *msgctxt, const char *msgid, +		  const char *msgid_plural, unsigned long int n, +		  int category) +{ +  size_t msgctxt_len = strlen (msgctxt) + 1; +  size_t msgid_len = strlen (msgid) + 1; +  const char *translation; +#if _LIBGETTEXT_HAVE_VARIABLE_SIZE_ARRAYS +  char msg_ctxt_id[msgctxt_len + msgid_len]; +#else +  char buf[1024]; +  char *msg_ctxt_id = +    (msgctxt_len + msgid_len <= sizeof (buf) +     ? buf +     : (char *) malloc (msgctxt_len + msgid_len)); +  if (msg_ctxt_id != NULL) +#endif +    { +      memcpy (msg_ctxt_id, msgctxt, msgctxt_len - 1); +      msg_ctxt_id[msgctxt_len - 1] = '\004'; +      memcpy (msg_ctxt_id + msgctxt_len, msgid, msgid_len); +      translation = dcngettext (domain, msg_ctxt_id, msgid_plural, n, category); +#if !_LIBGETTEXT_HAVE_VARIABLE_SIZE_ARRAYS +      if (msg_ctxt_id != buf) +	free (msg_ctxt_id); +#endif +      if (!(translation == msg_ctxt_id || translation == msgid_plural)) +	return translation; +    } +  return (n == 1 ? msgid : msgid_plural); +} + +#endif /* _LIBGETTEXT_H */ diff --git a/src/libr-backends.h b/src/libr-backends.h new file mode 100644 index 0000000..a0cd59c --- /dev/null +++ b/src/libr-backends.h @@ -0,0 +1,22 @@ +#ifndef __LIBR_BACKENDS_H +#define __LIBR_BACKENDS_H + +/* + * All of the backend functions are explicitly declared internal to prevent any custom backend + * from leaving out one of these critical functions. + */ +INTERNAL_FN libr_intstatus add_section(libr_file *file_handle, char *resource_name, libr_section **retscn); +INTERNAL_FN void *data_pointer(libr_section *scn, libr_data *data); +INTERNAL_FN size_t data_size(libr_section *scn, libr_data *data); +INTERNAL_FN libr_intstatus find_section(libr_file *file_handle, char *section, libr_section **retscn); +INTERNAL_FN libr_data *get_data(libr_file *file_handle, libr_section *scn); +INTERNAL_FN void initialize_backend(void); +INTERNAL_FN libr_data *new_data(libr_file *file_handle, libr_section *scn); +INTERNAL_FN libr_section *next_section(libr_file *file_handle, libr_section *scn); +INTERNAL_FN libr_intstatus remove_section(libr_file *file_handle, libr_section *scn); +INTERNAL_FN char *section_name(libr_file *file_handle, libr_section *scn); +INTERNAL_FN libr_intstatus set_data(libr_file *file_handle, libr_section *scn, libr_data *data, off_t offset, char *buffer, size_t size); +INTERNAL_FN libr_intstatus open_handles(libr_file *file_handle, char *filename, libr_access_t access); +INTERNAL_FN void write_output(libr_file *file_handle); + +#endif /* __LIBR_BACKENDS_H */ diff --git a/src/libr-bfd.c b/src/libr-bfd.c new file mode 100644 index 0000000..c4bc8b1 --- /dev/null +++ b/src/libr-bfd.c @@ -0,0 +1,533 @@ +/* + * + *  Copyright (c) 2008-2011 Erich Hoover + * + *  libr libbfd Backend - Add resources into ELF binaries using libbfd + *  + * *** PLEASE READ THE README FILE FOR LICENSE DETAILS SPECIFIC TO THIS FILE ***  + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser 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, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * 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 + * + * To provide feedback, report bugs, or otherwise contact me: + * ehoover at mines dot edu + * + */ + +/* Include compile-time parameters */ +#include "config.h" + +#include "libr.h" +#include "libr-internal.h" + +/* File access */ +#include <fcntl.h> + +/* Safe rename requires some errno() knowledge */ +#include <errno.h> + +#include <sys/stat.h> +#include <string.h> +#include <unistd.h> +#include <stdlib.h> +#include <stdio.h> + +/* + * Build the libr_file handle for processing with libbfd + */ +libr_intstatus open_handles(libr_file *file_handle, char *filename, libr_access_t access) +{ +	bfd *handle = NULL; +	 +	handle = bfd_openr(filename, "default"); +	if(!handle) +		RETURN(LIBR_ERROR_OPENFAILED, "Failed to open input file"); +	if(!bfd_check_format(handle, bfd_object)) +		RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: not a libbfd object"); +	if(bfd_get_flavour(handle) != bfd_target_elf_flavour) +		RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: not an ELF file"); +	bfd_set_error(bfd_error_no_error); +	file_handle->filename = filename; +	file_handle->bfd_read = handle; +	file_handle->access = access; +	if(access == LIBR_READ_WRITE) +	{ +		struct stat file_stat; +		int fd; +		 +		/* Check for write permission on the file */ +		fd = open(filename, O_WRONLY); +		if(fd == ERROR) +			RETURN(LIBR_ERROR_WRITEPERM, "No write permission for file"); +		close(fd); +		/* Obtain the access mode of the input file */ +		if(stat(filename, &file_stat) == ERROR) +			RETURN(LIBR_ERROR_NOSIZE, "Failed to obtain file size"); +		file_handle->filemode = file_stat.st_mode; +		file_handle->fileowner = file_stat.st_uid; +		file_handle->filegroup = file_stat.st_gid; +		/* Open a temporary file with the same settings as the input file */ +		strcpy(file_handle->tempfile, LIBR_TEMPFILE); +		file_handle->fd_handle = mkstemp(file_handle->tempfile); +		handle = bfd_openw(file_handle->tempfile, bfd_get_target(file_handle->bfd_read)); +		if(!bfd_set_format(handle, bfd_get_format(file_handle->bfd_read))) +			RETURN(LIBR_ERROR_SETFORMAT, "Failed to set output file format to input file format"); +		if(!bfd_set_arch_mach(handle, bfd_get_arch(file_handle->bfd_read), bfd_get_mach(file_handle->bfd_read))) +			RETURN(LIBR_ERROR_SETARCH, "Failed to set output file architecture to input file architecture"); +		/* twice needed ? */ +		if(!bfd_set_format(handle, bfd_get_format(file_handle->bfd_read))) +			RETURN(LIBR_ERROR_SETFORMAT, "Failed to set output file format to input file format"); +		file_handle->bfd_write = handle; +	} +	else +	{ +		file_handle->fd_handle = 0; +		file_handle->bfd_write = NULL; +	}  +	RETURN_OK; +} + +/* + * Check to see if a symbol should be kept + */ +int keep_symbol(libr_section *sections, libr_section *chkscn) +{ +	libr_section *scn; +	 +	/* Check that the section is publicly exposed */ +	for(scn = sections; scn != NULL; scn = scn->next) +	{ +		if(scn == chkscn) +		{ +			/* if it is, and has size zero, then it was marked for deletion */ +			if(bfd_get_section_size(chkscn) == 0) +				return false; +			return true; +		} +	} +	return true; +} + +/* + * Remove the symbol corresponding to a deleted section + */ +void remove_sections(libr_section *sections, void *symtab_buffer, long *symtab_count) +{  +	asymbol **symtab = (asymbol **) symtab_buffer; +	long i, cnt = *symtab_count; +	 +	for(i=0;i<cnt;i++) +	{ +		libr_section *chkscn = NULL; +		asymbol *symbol = symtab[i]; +		 +		if(symbol != NULL) +			chkscn = bfd_get_section(symbol); +		if(chkscn != NULL && !keep_symbol(sections, chkscn)) +		{ +			/* remove the symbol from the table */ +			asymbol **tmp = (asymbol **) malloc(sizeof(asymbol *) * (cnt-(i+1))); +			memcpy(&tmp[0], &symtab[i+1], sizeof(asymbol *) * (cnt-(i+1))); +			memcpy(&symtab[i], &tmp[0], sizeof(asymbol *) * (cnt-(i+1))); +			free(tmp); +			cnt--; +		} +	} +	*symtab_count = cnt; +} + +int setup_sections(bfd *ihandle, bfd *ohandle) +{ +	libr_section *iscn, *oscn; +	bfd_vma vma; +	 +	for(iscn = ihandle->sections; iscn != NULL; iscn = iscn->next) +	{ +		if(bfd_get_section_size(iscn) == 0) +		{ +			continue; /* Section has been marked for deletion */ +		} +		/* Use SEC_LINKER_CREATED to ask the libbfd backend to take care of configuring the section */ + +		// Keep the ARM_ATTRIBUTES section type intact on armhf systems +		// If this is not done, readelf -A will not print any architecture information for the modified library, +		// and ldd will report that the library cannot be found +		if (strcmp(iscn->name, ".ARM.attributes") == 0) +		{ +			oscn = bfd_make_section_anyway_with_flags(ohandle, iscn->name, iscn->flags); +		} +		else +		{ +			oscn = bfd_make_section_anyway_with_flags(ohandle, iscn->name, iscn->flags | SEC_LINKER_CREATED); +		} +		if(oscn == NULL) +		{ +			printf("failed to create out section: %s\n", bfd_errmsg(bfd_get_error())); +			return false; +		} +		if(!bfd_set_section_size(ohandle, oscn, iscn->size)) +		{ +			printf("failed to set data size: %s\n", bfd_errmsg(bfd_get_error())); +			return false; +		} +		vma = bfd_section_vma(ihandle, iscn); +		if(!bfd_set_section_vma(ohandle, oscn, vma)) +		{ +			printf("failed to set virtual memory address: %s\n", bfd_errmsg(bfd_get_error())); +			return false; +		} +		oscn->lma = iscn->lma; +		if(!bfd_set_section_alignment(ohandle, oscn, bfd_section_alignment(ihandle, iscn))) +		{ +			printf("failed to compute section alignment: %s\n", bfd_errmsg(bfd_get_error())); +			return false; +		} +		oscn->entsize = iscn->entsize; +		iscn->output_section = oscn; +		iscn->output_offset = vma; +		if(!bfd_copy_private_section_data(ihandle, iscn, ohandle, oscn)) +		{ +			printf("failed to compute section alignment: %s\n", bfd_errmsg(bfd_get_error())); +			return false; +		} +	} +	return true; +} + +/* + * Go through the rather complicated process of using libbfd to build the output file + */ +int build_output(libr_file *file_handle) +{ +	void *symtab_buffer = NULL, *reloc_buffer = NULL, *buffer = NULL; +	bfd_size_type symtab_size, reloc_size, size; +	bfd *ohandle = file_handle->bfd_write; +	bfd *ihandle = file_handle->bfd_read; +	long symtab_count, reloc_count; +	libr_section *iscn, *oscn; +	 +	if(!bfd_set_start_address(ohandle, bfd_get_start_address(ihandle))) +	{ +		printf("failed to set start address: %s\n", bfd_errmsg(bfd_get_error())); +		return false; +	} +	if(!bfd_set_file_flags(ohandle, bfd_get_file_flags(ihandle))) +	{ +		printf("failed to set file flags: %s\n", bfd_errmsg(bfd_get_error())); +		return false; +	} +	/* Setup the sections in the output file */ +	if(!setup_sections(ihandle, ohandle)) +		return false; /* error already printed */ +	if(!bfd_copy_private_header_data(ihandle, ohandle)) +	{ +		printf("failed to copy header: %s\n", bfd_errmsg(bfd_get_error())); +		return false; /* failed to create section */ +	} +	/* Get the old symbol table */ +	if((bfd_get_file_flags(ihandle) & HAS_SYMS) == 0) +	{ +		printf("file has no symbol table: %s\n", bfd_errmsg(bfd_get_error())); +		return false; +	} +	symtab_size = bfd_get_symtab_upper_bound(ihandle); +	if((signed)symtab_size < 0) +	{ +		printf("failed to get symbol table size: %s\n", bfd_errmsg(bfd_get_error())); +		return false; +	} +	symtab_buffer = malloc(symtab_size); +	symtab_count = bfd_canonicalize_symtab(ihandle, symtab_buffer); +	if(symtab_count < 0) +	{ +		printf("failed to get symbol table number of entries: %s\n", bfd_errmsg(bfd_get_error())); +		return false; +	} +	/* Tweak the symbol table to remove sections that no-longer exist */ +	remove_sections(ihandle->sections, symtab_buffer, &symtab_count); +	bfd_set_symtab(ohandle, symtab_buffer, symtab_count); +	/* Actually copy section data */ +	for(iscn = ihandle->sections; iscn != NULL; iscn = iscn->next) +	{ +		size = bfd_get_section_size(iscn); +		if(size == 0) +			continue; /* Section has been marked for deletion */ +		oscn = iscn->output_section; +		reloc_size = bfd_get_reloc_upper_bound(ihandle, iscn); +		if(reloc_size == 0) +			bfd_set_reloc(ohandle, oscn, NULL, 0); +		else +		{ +			reloc_buffer = malloc(reloc_size); +			reloc_count = bfd_canonicalize_reloc(ihandle, iscn, reloc_buffer, symtab_buffer); +			bfd_set_reloc(ohandle, oscn, reloc_buffer, reloc_count); +		} + +		if(bfd_get_section_flags(ihandle, iscn) & SEC_HAS_CONTENTS) +		{ +			/* NOTE: if the section is just being copied then do that, otherwise grab +			 * the user data for the section (stored previously by set_data) +			 */ +			if(iscn->userdata == NULL) +			{ +				buffer = malloc(size); +				if(!bfd_get_section_contents(ihandle, iscn, buffer, 0, size)) +				{ +					printf("failed to get section contents: %s\n", bfd_errmsg(bfd_get_error())); +					free(buffer); +					return false; +				} +			} +			else +				buffer = iscn->userdata; +			if(!bfd_set_section_contents(ohandle, oscn, buffer, 0, size)) +			{ +				printf("failed to set section contents: %s\n", bfd_errmsg(bfd_get_error())); +				free(buffer); +				return false; +			} +			free(buffer); +			if(!bfd_copy_private_section_data(ihandle, iscn, ohandle, oscn)) +			{ +				printf("failed to copy private section data: %s\n", bfd_errmsg(bfd_get_error())); +				return false; +			} +		} +	} +	if(!bfd_copy_private_bfd_data(ihandle, ohandle)) +	{ +		printf("failed to copy private data: %s\n", bfd_errmsg(bfd_get_error())); +		return false; +	} +	return true; +} + +/* + * Perform a cross-device safe rename + */ +int safe_rename(const char *old, const char *new) +{ +	char buffer[1024]; +	FILE *in, *out; +	int read; +	 +	in = fopen(old, "r"); +	if(!in) +		return -1; +	out = fopen(new, "w"); +	if(!out) +		return -1; +	while(!feof(in) && !ferror(in)) +	{ +		read = fread(buffer, 1, sizeof(buffer), in); +		fwrite(buffer, read, 1, out); +	} +	fclose(in); +	fclose(out); +	if(ferror(in)) +	{ +		remove(new); +		return -1; +	} +	return remove(old); +} + +/* + * Write the output file using the libbfd method + */  +void write_output(libr_file *file_handle) +{ +	int write_ok = false; +	 +	if(file_handle->bfd_write != NULL) +	{ +		write_ok = true; +		if(!build_output(file_handle)) +		{ +			printf("failed to build output file.\n"); +			write_ok = false; +		} +		if(!bfd_close(file_handle->bfd_write)) +		{ +			printf("failed to close write handle.\n"); +			write_ok = false; +		} +		if(file_handle->fd_handle != 0 && close(file_handle->fd_handle)) +		{ +			write_ok = false; +			printf("failed to close write file descriptor.\n"); +		} +	} +	/* The read handle must be closed last since it is used in the write process */  +	if(!bfd_close(file_handle->bfd_read)) +		printf("failed to close read handle.\n"); +	/* Copy the temporary output over the input */ +	if(write_ok) +	{ +		if(rename(file_handle->tempfile, file_handle->filename) < 0) +		{ +			if(errno != EXDEV || safe_rename(file_handle->tempfile, file_handle->filename) < 0) +				printf("failed to rename output file: %m\n"); +		} +		if(chmod(file_handle->filename, file_handle->filemode) < 0) +			printf("failed to set file mode.\n"); +		if(chown(file_handle->filename, file_handle->fileowner, file_handle->filegroup) < 0) +			printf("failed to set file ownership.\n"); +	} +} + +/* + * Find a named section from the ELF file using libbfd + */ +libr_intstatus find_section(libr_file *file_handle, char *section_name, libr_section **retscn) +{ +	libr_section *scn; +	 +	for(scn = file_handle->bfd_read->sections; scn != NULL; scn = scn->next) +	{ +		if(strcmp(scn->name, section_name) == 0) +		{ +			*retscn = scn; +			RETURN_OK; +		} +	} +	RETURN(LIBR_ERROR_NOSECTION, "ELF resource section not found"); +} + +/* + * Obtain the data from a section using libbfd + */ +libr_data *get_data(libr_file *file_handle, libr_section *scn) +{ +	libr_data *data = malloc(scn->size); +	 +	if(!bfd_get_section_contents(file_handle->bfd_read, scn, data, 0, scn->size)) +	{ +		free(data); +		data = NULL; +	} +	scn->userdata = data; +	return data; +} + +/* + * Create new data for a section using libbfd + */ +libr_data *new_data(libr_file *file_handle, libr_section *scn) +{ +	/* NOTE: expanding data is handled by set_data for libbfd */ +	if(scn->userdata != NULL) +		return scn->userdata; +	scn->size = 0; +	scn->userdata = malloc(0); +	return scn->userdata; +} + +/* + * Create new data for a section using libbfd (at least, do so memory-wise) + */ +libr_intstatus set_data(libr_file *file_handle, libr_section *scn, libr_data *data, off_t offset, char *buffer, size_t size) +{ +	char *intbuffer = NULL; +	 +	/* special case: clear buffer */ +	if(buffer == NULL) +	{ +		scn->size = 0; +		if(scn->userdata != NULL) +			free(scn->userdata); +		RETURN_OK; +	} +	/* normal case: add new data to the buffer */ +	scn->size = offset + size; +	scn->userdata = realloc(data, scn->size); +	if(scn->userdata == NULL) +		RETURN(LIBR_ERROR_MEMALLOC, "Failed to allocate memory for data"); +	intbuffer = scn->userdata; +	memcpy(&intbuffer[offset], buffer, size); +	RETURN_OK; +} + +/* + * Create a new section using libbfd + */ +libr_intstatus add_section(libr_file *file_handle, char *resource_name, libr_section **retscn) +{ +	libr_section *scn = NULL; +	 +	scn = bfd_make_section(file_handle->bfd_read, resource_name); +	if(scn == NULL) +		RETURN(LIBR_ERROR_NEWSECTION, "Failed to create new section"); +	if(!bfd_set_section_flags(file_handle->bfd_read, scn, SEC_HAS_CONTENTS | SEC_DATA | SEC_IN_MEMORY)) +		RETURN(LIBR_ERROR_SETFLAGS, "Failed to set flags for section"); +	*retscn = scn; +	RETURN_OK; +} + +/* + * Remove a section and eliminate it from the ELF string table using libbfd + */ +libr_intstatus remove_section(libr_file *file_handle, libr_section *scn) +{ +	scn->size = 0; +	RETURN_OK; +} + +/* + * Return the pointer to the actual data in the section + */ +void *data_pointer(libr_section *scn, libr_data *data) +{  +	return data; +} + +/* + * Return the size of the data in the section + */ +size_t data_size(libr_section *scn, libr_data *data) +{ +	return scn->size; +} + +/* + * Return the next section in the ELF file + */ +libr_section *next_section(libr_file *file_handle, libr_section *scn) +{ +	/* get the first section */ +	if(scn == NULL) +	{ +		if(file_handle->bfd_read == NULL) +			return NULL; +		return file_handle->bfd_read->sections; +	} +	return scn->next; +} + +/* + * Return the name of a section + */ +char *section_name(libr_file *file_handle, libr_section *scn) +{ +	return (char *) scn->name; +} + +/* + * Initialize libbfd + */ +void initialize_backend(void) +{ +	bfd_init(); +} + diff --git a/src/libr-bfd.h b/src/libr-bfd.h new file mode 100644 index 0000000..7b6ea3e --- /dev/null +++ b/src/libr-bfd.h @@ -0,0 +1,40 @@ +#ifndef __LIBR_BFD_H +#define __LIBR_BFD_H + +#include "config.h" + +#include <sys/types.h> +#include <stdint.h> +#include <bfd.h> + +#if BFD_HOST_64BIT_LONG +	#if defined(__i386) +		#error "Using incorrect binutils header file for architecture." +	#endif +#else +	#if defined(__amd64) +		#error "Using incorrect binutils header file for architecture." +	#endif +#endif + +#ifndef DOXYGEN_SHOULD_SKIP_THIS + +typedef struct _libr_file { +	int fd_handle; +	bfd *bfd_read; +	bfd *bfd_write; +	char *filename; +	mode_t filemode; +	uid_t fileowner; +	gid_t filegroup; +	char tempfile[LIBR_TEMPFILE_LEN]; +	libr_access_t access; +} libr_file; + +#endif /* DOXYGEN_SHOULD_SKIP_THIS */ + +/* for a clean internal API */ +typedef asection libr_section; +typedef void libr_data; + +#endif /* __LIBR_BFD_H */ diff --git a/src/libr-elf.c b/src/libr-elf.c new file mode 100644 index 0000000..be2daae --- /dev/null +++ b/src/libr-elf.c @@ -0,0 +1,412 @@ +/* + * + *  Copyright (c) 2008 Erich Hoover + * + *  libr libelf Backend - Add resources into ELF binaries using libelf + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2.1 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * 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 + * + * To provide feedback, report bugs, or otherwise contact me: + * ehoover at mines dot edu + * + */ + +/* Include compile-time parameters */ +#include "config.h" + +#include "libr.h" + +#include <sys/stat.h> +#include <string.h> +#include <unistd.h> +#include <stdlib.h> +#include <stdio.h> +#include <fcntl.h> + +//#define MANUAL_LAYOUT            true + +extern void libr_set_error(libr_intstatus error); + +/* + * Write the output file using libelf + */ +void write_output(libr_file *file_handle) +{ +	/* Update the ELF file on the disk */ +	if(elf_update(file_handle->elf_handle, ELF_C_NULL) < 0) +	{ +		printf("elf_update() failed: %s.", elf_errmsg(-1)); +		return; +	} +	if(elf_update(file_handle->elf_handle, ELF_C_WRITE) < 0) +	{ +		printf("elf_update() failed: %s.", elf_errmsg(-1)); +		return; +	} +	/* Close the handles */ +	elf_end(file_handle->elf_handle); +	close(file_handle->fd_handle); +} + +/* + * Return the size of the file represented by the file descriptor + */ +off_t file_size(int fd) +{ +	struct stat file_stat; +	 +	if(fstat(fd, &file_stat) == ERROR) +		return ERROR; +	return file_stat.st_size; +} + +/* + * Open the handles for working with the file using libelf + */ +libr_intstatus open_handles(libr_file *file_handle, char *filename, libr_access_t access) +{ +	const int elf_access[2] = {ELF_C_READ, ELF_C_RDWR}; +	const int fd_access[2] = {O_RDONLY, O_RDWR}; +	Elf *e = NULL; +	int fd = 0; +	 +	if((fd = open(filename, fd_access[access], 0)) < 0) +		RETURN(LIBR_ERROR_OPENFAILED, "Failed to open input file"); +	if((e = elf_begin(fd, elf_access[access], NULL)) == NULL) +		RETURN(LIBR_ERROR_BEGINFAILED, "Failed to open ELF file: %s.", elf_errmsg(-1)); +	if(elf_kind(e) != ELF_K_ELF) +		RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format"); +	 +	file_handle->access = access; +	file_handle->fd_handle = fd; +	file_handle->elf_handle = e; +	file_handle->file_size = file_size(fd); +	file_handle->version = EV_CURRENT; /* This should probably match the rest of the file */ +	RETURN_OK; +} + +/* + * Expand a section + * (Only used when manually controlling ELF layout) + */ +#ifdef MANUAL_LAYOUT +libr_intstatus expand_section(Elf *e, Elf_Scn *scn, size_t size, int reset) +{ +	size_t offset = 0, delta = 0; +	Elf_Scn *tmpscn = NULL; +	GElf_Shdr shdr; +	 +	if(gelf_getshdr(scn, &shdr) != &shdr) +		RETURN(LIBR_INTERROR_GETSHDR, "Failed to obtain ELF header: %s", elf_errmsg(-1)); +	if(reset) +	{ +		delta = (size-shdr.sh_size);  +		shdr.sh_size = size; +	} +	else +	{ +		delta = size; +		shdr.sh_size += size; +	} +	offset = shdr.sh_offset; +	if(gelf_update_shdr(scn, &shdr) < 0) +		RETURN(LIBR_ERROR_UPDATE, "Failed to perform dynamic update: %s.", elf_errmsg(-1)); +	if(elf_update(e, ELF_C_NULL) < 0) +		RETURN(LIBR_ERROR_UPDATE, "Failed to perform dynamic update: %s.", elf_errmsg(-1)); +	/* Update any section that follows this one data-wise */ +/* +****** This does not work yet +	while((tmpscn = elf_nextscn(e, tmpscn)) != NULL) +	{ +		if(tmpscn == scn) +			continue; +		if(gelf_getshdr(tmpscn, &shdr) != &shdr) +			return LIBR_INTERROR_GETSHDR; +		if(offset < shdr.sh_offset) +		{ +			if((name = elf_strptr(e, ehdr.e_shstrndx, shdr.sh_name)) == NULL) +				RETURN(LIBR_ERROR_STRPTR, "Failed to obtain section string pointer: %s.", elf_errmsg(-1)); +			shdr.sh_offset += delta; +			if(gelf_update_shdr(tmpscn, &shdr) < 0) +				RETURN(LIBR_ERROR_UPDATE, "Failed to perform dynamic update: %s.", elf_errmsg(-1)); +			if(elf_update(e, ELF_C_NULL) < 0) +				RETURN(LIBR_ERROR_UPDATE, "Failed to perform dynamic update: %s.", elf_errmsg(-1)); +		} +	} +*/ +	return LIBR_OK; +} +#endif /* MANUAL_LAYOUT */ + +/* + * Obtain the data from a section using libelf + */ +libr_data *get_data(libr_file *file_handle, libr_section *scn) +{ +	return elf_getdata(scn, NULL); +} + +/* + * Create new data for a section using libelf + */ +libr_data *new_data(libr_file *file_handle, libr_section *scn) +{ +	return elf_newdata(scn); +} + +/* + * Set data for a section using libelf (not written yet) + */ +libr_intstatus set_data(libr_file *file_handle, libr_section *scn, libr_data *data, off_t offset, char *buffer, size_t size) +{ +	data->d_align = 1; +	data->d_off = offset; +	data->d_buf = buffer; +	data->d_type = ELF_T_BYTE; +	data->d_size = size; +	data->d_version = file_handle->version; +#ifdef MANUAL_LAYOUT +	if(expand_section(file_handle->elf_handle, scn, data->d_size, true) != LIBR_OK) +		RETURN(LIBR_ERROR_EXPANDSECTION, "Failed to expand section"); +#else +	if(elf_update(file_handle->elf_handle, ELF_C_NULL) < 0) +		RETURN(LIBR_ERROR_UPDATE, "Failed to perform dynamic update: %s.", elf_errmsg(-1)); +	if(elf_update(file_handle->elf_handle, ELF_C_WRITE) < 0) +		RETURN(LIBR_ERROR_UPDATE, "Failed to perform dynamic update: %s.", elf_errmsg(-1)); +#endif /* MANUAL_LAYOUT */ +	RETURN_OK; +} + +/* + * Find a named section from the ELF file using libelf + */ +libr_intstatus find_section(libr_file *file_handle, char *section, libr_section **retscn) +{ +	Elf *e = file_handle->elf_handle; +	Elf_Scn *scn = NULL; +	char *name = NULL; +	GElf_Ehdr ehdr; +	GElf_Shdr shdr; +	uintmax_t si; +	 +	if(gelf_getehdr(e, &ehdr) == NULL) +		RETURN(LIBR_ERROR_GETEHDR, "Failed to obtain ELF header: %s", elf_errmsg(-1)); +	while((scn = elf_nextscn(e, scn)) != NULL) +	{ +		if(gelf_getshdr(scn, &shdr) != &shdr) +			RETURN(LIBR_ERROR_GETSHDR, "Failed to obtain ELF section header: %s", elf_errmsg(-1)); +		if((name = elf_strptr(e, ehdr.e_shstrndx, shdr.sh_name)) == NULL) +			RETURN(LIBR_ERROR_STRPTR, "Failed to obtain section string pointer: %s.", elf_errmsg(-1)); +		 +		si = (uintmax_t) elf_ndxscn(scn); +/* +printf("%d: %s (%d %d)\n", (long) si, name, (long) shdr.sh_offset, (long) shdr.sh_size); +*/ +		if(strcmp(name, section) == 0) +		{ +			*retscn = scn; +			RETURN_OK; +		} +	} +	RETURN(LIBR_ERROR_NOSECTION, "ELF resource section not found"); +} + +/* + * Add a new section and create a name for it in the ELF string table using libelf + */ +libr_intstatus add_section(libr_file *file_handle, char *section, Elf_Scn **retscn) +{ +	Elf_Scn *scn = NULL, *strscn = NULL; +	Elf *e = file_handle->elf_handle; +#ifdef MANUAL_LAYOUT +	size_t tblsize = 0; +#endif /* MANUAL_LAYOUT */ +	Elf_Data *data; +	GElf_Ehdr ehdr; +	GElf_Shdr shdr; +	 +	if(gelf_getehdr(e, &ehdr) == NULL) +		RETURN(LIBR_ERROR_GETEHDR, "Failed to obtain ELF header: %s", elf_errmsg(-1)); +	/* TODO: Support creating a string table for objects that don't have one */ +	if(!ehdr.e_shstrndx) +		RETURN(LIBR_ERROR_NOTABLE, "No ELF string table"); +	strscn = elf_getscn(e, ehdr.e_shstrndx); +	if(strscn == NULL)  +		RETURN(LIBR_ERROR_TABLE, "Failed to open string table: %s.", elf_errmsg(-1)); +	data = elf_newdata(strscn); +	if(data == NULL) +		RETURN(LIBR_ERROR_NEWDATA, "Failed to create data for section"); +	data->d_align = 1; + +#ifdef MANUAL_LAYOUT +{ +	GElf_Shdr strshdr; +	 +	if(gelf_getshdr(strscn, &strshdr) != &strshdr) +		RETURN(LIBR_ERROR_GETSHDR, "Failed to obtain ELF section header: %s", elf_errmsg(-1)); +	data->d_off = strshdr.sh_size; +#endif /* MANUAL_LAYOUT */ + +	data->d_size = (size_t) strlen(section)+1; +	data->d_type = ELF_T_BYTE; +	data->d_buf = section; +	data->d_version = file_handle->version; + +#ifdef MANUAL_LAYOUT +	if(expand_section(e, strscn, data->d_size, false) != LIBR_OK) +		return false; +} +#else +	/* Update the internal offset information */ +	if(elf_update(e, ELF_C_NULL) < 0) +		RETURN(LIBR_ERROR_UPDATE, "Failed to perform dynamic update: %s.", elf_errmsg(-1)); +#endif /* MANUAL_LAYOUT */ + +	/* seek to the end of the section data */ +	if((scn = elf_newscn(e)) == NULL) +		RETURN(LIBR_ERROR_NEWSECTION, "Failed to create new section"); +	if(gelf_getshdr(scn, &shdr) != &shdr) +		RETURN(LIBR_ERROR_GETSHDR, "Failed to obtain ELF section header: %s", elf_errmsg(-1)); +	shdr.sh_addralign = 1; +#ifdef MANUAL_LAYOUT +	shdr.sh_offset = file_handle->file_size; +#endif /* MANUAL_LAYOUT */ +	shdr.sh_size = 0; +	shdr.sh_name = data->d_off; +	shdr.sh_type = SHT_NOTE; /* TODO: Does "NOTE" type fit best? */ +	shdr.sh_flags = SHF_WRITE; +	shdr.sh_entsize = 0; +	if(gelf_update_shdr(scn, &shdr) < 0) +		RETURN(LIBR_ERROR_UPDATE, "Failed to perform dynamic update: %s.", elf_errmsg(-1)); +	*retscn = scn; +	RETURN_OK; +} + +/* + * Remove a section and eliminate it from the ELF string table using libelf + */ +libr_intstatus remove_section(libr_file *file_handle, libr_section *scn) +{ +	unsigned int table_size, str_size; +	char *buffer = NULL, *tmp = NULL; +	Elf *e = file_handle->elf_handle; +	int remaining_size; +	Elf_Scn *strscn; +	Elf_Data *data; +	GElf_Ehdr ehdr; +	GElf_Shdr shdr; +	 +	if(gelf_getehdr(e, &ehdr) == NULL) +		RETURN(LIBR_ERROR_GETEHDR, "Failed to obtain ELF header: %s", elf_errmsg(-1)); +	/* Grab the string table */ +	if(!ehdr.e_shstrndx) +		RETURN(LIBR_ERROR_NOTABLE, "No ELF string table"); +	strscn = elf_getscn(e, ehdr.e_shstrndx); +	if(strscn == NULL) +		RETURN(LIBR_ERROR_TABLE, "Failed to open string table: %s.", elf_errmsg(-1)); +	if((data = elf_getdata(strscn, NULL)) == NULL) +		RETURN(LIBR_ERROR_GETDATA, "Failed to obtain data of section"); +	/* Find where the section name is in the string table */ +	if(gelf_getshdr(scn, &shdr) != &shdr) +		RETURN(LIBR_ERROR_GETSHDR, "Failed to obtain ELF section header: %s", elf_errmsg(-1)); +	table_size = data->d_size; +	buffer = (char *) data->d_buf; +	/* Excise the string from the table */ +	str_size = strlen(&buffer[shdr.sh_name])+1; +	remaining_size = table_size-(shdr.sh_name+str_size); +	if(remaining_size < 0) +		RETURN(LIBR_ERROR_SIZEMISMATCH, "Section's data size does not make sense"); +	if(remaining_size > 0) +	{ +		/* If there is data after our icon entry in the table then it must be moved before resizing +		 * NOTE: Using memcpy with overlapping addresses is not allowed, use temporary buffer.  +		 */ +		tmp = (char *) malloc(remaining_size); +		memcpy(tmp, &buffer[shdr.sh_name+str_size], remaining_size); +		memcpy(&buffer[shdr.sh_name], tmp, remaining_size); +		free(tmp); +	} +	data->d_size -= str_size; +	/* Update the internal offset information */ +	if(elf_update(e, ELF_C_NULL) < 0) +		RETURN(LIBR_ERROR_UPDATE, "Failed to perform dynamic update: %s.", elf_errmsg(-1)); +#ifdef MANUAL_LAYOUT +{ +	GElf_Shdr strshdr; +	 +	if(gelf_getshdr(strscn, &strshdr) != &strshdr) +		RETURN(LIBR_ERROR_GETSHDR, "Failed to obtain ELF section header: %s", elf_errmsg(-1)); +	strshdr.sh_size -= str_size; +	if(gelf_update_shdr(strscn, &strshdr) < 0) +		RETURN(LIBR_ERROR_UPDATE, "Failed to perform dynamic update: %s.", elf_errmsg(-1)); +} +#endif /* MANUAL_LAYOUT */ + +	/* Clear the section itself and update the offsets */ +	if(elfx_remscn(e, scn) == 0) +		RETURN(LIBR_ERROR_REMOVESECTION, "Failed to remove section: %s.", elf_errmsg(-1)); +	RETURN_OK; +} + +/* + * Return the pointer to the actual data in the section + */ +void *data_pointer(libr_section *scn, libr_data *data) +{  +	return data->d_buf; +} + +/* + * Return the size of the data in the section + */ +size_t data_size(libr_section *scn, libr_data *data) +{ +	return data->d_size; +} + +/* + * Return the next section in the ELF file + */ +libr_section *next_section(libr_file *file_handle, libr_section *scn) +{ +	return elf_nextscn(file_handle->elf_handle, scn); +} + +/* + * Retrieve the name of a section + */ +char *section_name(libr_file *file_handle, libr_section *scn) +{ +	char *name = NULL; +	GElf_Shdr shdr; +	GElf_Ehdr ehdr; +	 +	if(gelf_getehdr(file_handle->elf_handle, &ehdr) == NULL) +		return NULL; +	if(gelf_getshdr(scn, &shdr) != &shdr) +		return NULL; +	if((name = elf_strptr(file_handle->elf_handle, ehdr.e_shstrndx, shdr.sh_name)) == NULL) +		return NULL; +	return strdup(name); +} + +/* + * Initialize the libelf backend + */ +void initialize_backend(void) +{ +	if(elf_version(EV_CURRENT) == EV_NONE) +		return; //errx(EX_SOFTWARE, "ELF library initialization failed: %s", elf_errmsg(-1)); +} diff --git a/src/libr-elf.h b/src/libr-elf.h new file mode 100644 index 0000000..4c632e8 --- /dev/null +++ b/src/libr-elf.h @@ -0,0 +1,24 @@ +#ifndef __LIBR_ELF_H +#define __LIBR_ELF_H + +/* Handle ELF files */ +#include <libelf.h> +#include <gelf.h> + +#ifndef DOXYGEN_SHOULD_SKIP_THIS + +typedef struct _libr_file { +	int fd_handle; +	Elf *elf_handle; +	size_t file_size; +	libr_access_t access; +	unsigned int version; +} libr_file; + +#endif /* DOXYGEN_SHOULD_SKIP_THIS */ + +/* for a clean internal API */ +typedef Elf_Scn libr_section; +typedef Elf_Data libr_data; + +#endif /* __LIBR_ELF_H */ diff --git a/src/libr-gtk.c b/src/libr-gtk.c new file mode 100644 index 0000000..f746aa8 --- /dev/null +++ b/src/libr-gtk.c @@ -0,0 +1,443 @@ +/* + * + *  Copyright (c) 2008-2009 Erich Hoover + * + *  libr GTK support - Convenience functions for using resources in GTK applications + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2.1 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * 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 + * + * To provide feedback, report bugs, or otherwise contact me: + * ehoover at mines dot edu + * + */ + +/* Include compile-time parameters */ +#include "config.h" + +#include "libr.h" +#include "libr-gtk.h" +#include "libr-icons.h" +#include "tempfiles.h" + +/* For loading GTK/GDK images */ +#include <gdk-pixbuf/gdk-pixbuf.h> +#include <glib.h> + +/* For loading GLADE files */ +#include <glade/glade.h> + +/* For loading GTK+ Builder files */ +#include <gtk/gtk.h> + +/* For malloc/free */ +#include <stdlib.h> + +/* For string handling */ +#include <string.h> + +typedef gchar * (*GladeFileCallback)(GladeXML *, const gchar *, guint *); +GladeFileCallback glade_set_file_callback(GladeFileCallback callback, gpointer user_data); + +/* Use weak binding for all glade and GTK+ requirements */ +#pragma weak glade_set_file_callback + +#pragma weak gtk_window_set_default_icon_list +#pragma weak gdk_pixbuf_loader_get_pixbuf +#pragma weak gtk_builder_add_from_string +#pragma weak gdk_pixbuf_loader_set_size +#pragma weak g_type_check_instance_cast +#pragma weak gtk_builder_add_from_file +#pragma weak glade_xml_new_from_buffer +#pragma weak gdk_pixbuf_loader_write +#pragma weak gdk_pixbuf_loader_close +#pragma weak gdk_pixbuf_loader_new +#pragma weak g_signal_connect_data +#pragma weak g_signal_connect +#pragma weak gtk_builder_new +#pragma weak g_object_unref +#pragma weak glade_xml_new +#pragma weak g_list_append +#pragma weak glade_init +#pragma weak gtk_init +#pragma weak g_free + +#define GLADE_SECTION   ".glade" +#define BUILDER_SECTION ".ui" + +/* + * Handle the resource request from libglade + */ +gchar *libr_glade_read_resource(GladeXML *gladefile, const gchar *filename, guint *size, gpointer user_data) +{ +	return libr_malloc((libr_file *) user_data, (char *) filename, (size_t *) size);  +} + +/* + * Handle the resource request from GtkBuilder + */ +gboolean libr_gtk_read_resource(GtkBuilder *builder, const gchar *filename, gchar **data, gsize *size, GError **error, gpointer user_data) +{ +	if(data == NULL) +		return FALSE; +	*data = libr_malloc((libr_file *) user_data, (char *) filename, (size_t *) size); +	if(*data == NULL) +		return FALSE; +	return TRUE; +} + +/* + * Load the libglade resource appropriately for the currently installed version + * (AKA, hurray hacks!) + */ +GladeXML *libr_new_glade(libr_file *handle, char *gladefile, size_t gladefile_size) +{ +	if(glade_set_file_callback) /* The not-yet (ever?) existing way */ +	{ +		/* Register a callback for libglade to load our resources */ +		if(glade_set_file_callback((GladeFileCallback) libr_glade_read_resource, handle) != NULL) +			printf("warning: over-wrote an application callback!\n"); +		/* Initialize libglade almost as usual, just use a buffer instead of a file */ +		return glade_xml_new_from_buffer(gladefile, gladefile_size, NULL, NULL); +	} +	else /* The hacky way */ +	{ +		char *glade_file[PATH_MAX]; +		GladeXML *ret = NULL; +		char *temp_folder; +		 +		temp_folder = libr_extract_resources(handle); +		if(temp_folder == NULL) +			return NULL; +		strcpy((char*)glade_file, temp_folder); +		strcat((char*)glade_file, "/"); +		strcat((char*)glade_file, GLADE_SECTION); +		ret = glade_xml_new((char*)glade_file, NULL, NULL); +		if(ret == NULL) +			cleanup_folder(temp_folder); +		else +			register_folder_cleanup(temp_folder); +		return ret; +	} +} + +/* + * Load the GtkBuilder resource appropriately for the currently installed version + * (AKA, hurray hacks!) + */ +int libr_new_builder(libr_file *handle, char *builderfile, size_t builderfile_size, GtkBuilder *builder) +{ +	/* Register a callback for GtkBuilder to load our resources */ +	if(g_signal_connect(builder, "load-resource", (GCallback) libr_gtk_read_resource, handle)) +	{ +		/* Initialize GtkBuilder almost as usual, just use a buffer instead of a file */ +		if(gtk_builder_add_from_string(builder, builderfile, builderfile_size, NULL) == 0) +			return false; +		return true; +	} +	else /* The hacky way */ +	{ +		char *builder_file[PATH_MAX]; +		char *temp_folder; +		int ret = false; +		 +		temp_folder = libr_extract_resources(handle); +		if(temp_folder == NULL) +			return false; +		strcpy((char*)builder_file, temp_folder); +		strcat((char*)builder_file, "/"); +		strcat((char*)builder_file, BUILDER_SECTION); +		ret = gtk_builder_add_from_file(builder, (char*)builder_file, NULL); +		if(ret == 0) +			cleanup_folder(temp_folder); +		else +			register_folder_cleanup(temp_folder); +		g_free(temp_folder); +		return (ret != 0); +	} +} + +/* + * Return a GTK icon list + */ +EXPORT_FN IconList *libr_gtk_iconlist(libr_file *handle) +{ +	int sizes[] = {16, 32, 48, 96, 128}; +	IconList *icons = NULL; +	GdkPixbuf *icon = NULL; +	int sizes_len = 5, i; +	 +	if(handle == NULL) +	{ +		/* Must pass a file handle to obtain the icons from */ +		return NULL; +	} +	if(gtk_init == NULL) +	{ +		/* GTK+ was not linked with the application */ +		return false; +	} +	/* Go through the list of GTK "required" image sizes and build the icons */ +	for(i=0;i<sizes_len;i++) +	{  +		libr_icon *icon_handle = libr_icon_geticon_bysize(handle, sizes[i]); +		GdkPixbufLoader *stream = gdk_pixbuf_loader_new(); +		char *iconfile = NULL; +		size_t iconfile_size; +		 +		if(icon_handle == NULL) +		{ +			/* Failed to find an icon */ +			printf("Failed to find an icon\n"); +			continue; +		} +		iconfile = libr_icon_malloc(icon_handle, &iconfile_size); +		if(iconfile == NULL) +		{ +			/* Failed to obtain embedded icon */ +			continue; +		} +		libr_icon_close(icon_handle); +		/* TODO: Use the "size-prepared" signal to properly scale the width and height +void user_function (GdkPixbufLoader *loader, gint width, gint height, gpointer user_data) +		 */  +		gdk_pixbuf_loader_set_size(stream, sizes[i], sizes[i]); +		if(!gdk_pixbuf_loader_write(stream, (unsigned char *)iconfile, iconfile_size, NULL)) +		{ +			/* Failed to process image */ +			continue; +		} +		if(!gdk_pixbuf_loader_close(stream, NULL)) +		{ +			/* Failed to create image */ +			continue; +		} +		icon = gdk_pixbuf_loader_get_pixbuf(stream); +		if(icon == NULL) +		{ +			/* Failed to convert image to pixbuf */ +			continue; +		} +		icons = g_list_append(icons, icon); +	} +	return icons; +} + +/* + * Shared GtkBuilder resource loading + */ +GtkBuilder *libr_gtk_load_internal(libr_file *handle, char *resource_name) +{ +	GtkBuilder *builder = NULL; +	size_t builder_length; +	char *builder_data; +	 +	/* Obtain the GtkBuilder XML definition */ +	builder_data = libr_malloc(handle, resource_name, &builder_length); +	if(builder_data == NULL) +	{ +		/* Failed to obtain embedded GtkBuilder definition file */ +		goto failed; +	} +	/* Setup the GtkBuilder environment */ +	builder = gtk_builder_new(); +	if(builder == NULL) +		goto failed; +	if(!libr_new_builder(handle, builder_data, builder_length, builder)) +	{ +		/* Failed to build interface from resource file */ +		g_object_unref(G_OBJECT(builder)); +		goto failed; +	} +failed: +	free(builder_data); +	return builder; +} + +/* + * Load the requested GtkBuilder resource and any applicable icons + */ +EXPORT_FN int libr_gtk_load(BuilderHandle **gtk_ret, char *resource_name) +{ +	libr_file *handle; +	int ret = false; +	 +	if(gtk_ret == NULL) +	{ +		/* Why on earth would you call this without obtaining the handle to the resource? */ +		return false; +	} +	if(gtk_builder_new == NULL) +	{ +		/* GtkBuilder was not linked with the application */ +		return false; +	} +	/* Obtain the handle to the executable */ +	if((handle = libr_open(NULL, LIBR_READ)) == NULL) +	{ +		/* "Failed to open this executable (%s) for resources", progname() */ +		return false; +	} +	register_internal_handle(handle); +	*gtk_ret = libr_gtk_load_internal(handle, resource_name); +	if(*gtk_ret == NULL) +		goto failed; +	ret = true; +failed: +	if(!ret) +		libr_close(handle); +	return ret; +} + +/* + * Automatically load the ".ui" GtkBuilder resource and any applicable icons + */ +EXPORT_FN int libr_gtk_autoload(BuilderHandle **gtk_ret, IconList **icons_ret, int set_default_icon) +{ +	GList *icons = NULL; +	libr_file *handle; +	int ret = false; +	 +	if(gtk_ret == NULL) +	{ +		/* Why on earth would you call this without obtaining the handle to the resource? */ +		return false; +	} +	if(gtk_builder_new == NULL) +	{ +		/* GtkBuilder was not linked with the application */ +		return false; +	} +	/* Obtain the handle to the executable */ +	if((handle = libr_open(NULL, LIBR_READ)) == NULL) +	{ +		/* "Failed to open this executable (%s) for resources", progname() */ +		return false; +	} +	register_internal_handle(handle); +	/* Obtain the icons from the ELF binary */ +	icons = libr_gtk_iconlist(handle); +	/* Set the embedded icons as the default icon list (if requested) */ +	if(icons != NULL && set_default_icon) +		gtk_window_set_default_icon_list(icons); +	*gtk_ret = libr_gtk_load_internal(handle, BUILDER_SECTION); +	if(*gtk_ret == NULL) +		goto failed; +	if(icons_ret) +		*icons_ret = icons; +	ret = true; +failed: +	if(!ret) +		libr_close(handle); +	return ret; +} + +/* + * Shared libglade resource loading + */ +GladeXML *libr_glade_load_internal(libr_file *handle, char *resource_name) +{ +	GladeXML *glade = NULL; +	size_t glade_length; +	char *glade_data; +	 +	/* Obtain the GLADE XML definition */ +	glade_data = libr_malloc(handle, resource_name, &glade_length); +	if(glade_data == NULL) +	{ +		/* Failed to obtain embedded glade file */ +		goto failed; +	} +	/* Initialize libglade appropriate for the available version */ +	glade = libr_new_glade(handle, glade_data, glade_length); +	if(glade == NULL) +	{ +		/* Failed to initialize embedded glade file */ +		goto failed; +	} +failed: +	free(glade_data); +	return glade; +} + +/* + * Load the requested libglade resource and any applicable icons + */ +EXPORT_FN int libr_glade_load(GladeHandle **glade_ret, char *resource_name) +{ +	libr_file *handle; +	int ret = false; +	 +	if(glade_ret == NULL) +	{ +		/* Why on earth would you call this without obtaining the handle to the resource? */ +		return false; +	} +	if(glade_init == NULL) +	{ +		/* libglade was not linked with the application */ +		return false; +	} +	/* Obtain the handle to the executable */ +	if((handle = libr_open(NULL, LIBR_READ)) == NULL) +	{ +		/* "Failed to open this executable (%s) for resources", progname() */ +		return false; +	} +	register_internal_handle(handle); +	*glade_ret = libr_glade_load_internal(handle, resource_name); +	if(*glade_ret == NULL) +		goto failed; +	ret = true; +failed: +	if(!ret) +		libr_close(handle); +	return ret; +} + +/* + * Automatically load the ".glade" resource and any applicable icons + */ +EXPORT_FN int libr_glade_autoload(GladeHandle **glade_ret, IconList **icons_ret, int set_default_icon) +{ +	libr_file *handle = NULL; +	GList *icons = NULL; +	 +	if(glade_ret == NULL) +	{ +		/* Why on earth would you call this without obtaining the handle to the resource? */ +		return false; +	} +	if(glade_init == NULL) +	{ +		/* libglade was not linked with the application */ +		return false; +	} +	/* Obtain the handle to the executable */ +	if((handle = libr_open(NULL, LIBR_READ)) == NULL) +	{ +		/* "Failed to open this executable (%s) for resources", progname() */ +		return false; +	} +	register_internal_handle(handle); +	icons = libr_gtk_iconlist(handle); +	/* Set the embedded icons as the default icon list (if requested) */ +	if(icons != NULL && set_default_icon) +		gtk_window_set_default_icon_list(icons); +	/* Return the libglade and icon handles for the application */ +	*glade_ret = libr_glade_load_internal(handle, GLADE_SECTION); +	if(icons_ret) +		*icons_ret = icons; +	return true; +} diff --git a/src/libr-gtk.h b/src/libr-gtk.h new file mode 100644 index 0000000..fa6ba1b --- /dev/null +++ b/src/libr-gtk.h @@ -0,0 +1,55 @@ +/* + * + *  Copyright (c) 2008 Erich Hoover + * + *  libr-gtk - Convenience support for GTK+ + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2.1 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * 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 + * + * To provide feedback, report bugs, or otherwise contact me: + * ehoover at mines dot edu + * + */ + +#ifndef __LIBR_GTK_H +#define __LIBR_GTK_H + +#include "libr.h" + +#ifndef GLADE_H +	typedef void GladeHandle; +#else +	typedef GladeXML GladeHandle; +#endif +#ifndef __GTK_BUILDER_H__ +	typedef void BuilderHandle; +#else +	typedef GtkBuilder BuilderHandle; +#endif +#ifndef __G_LIB_H__ +	typedef void IconList; +#else +	typedef GList IconList; +#endif + +/* GTK Convenience API */ +IconList *libr_gtk_iconlist(libr_file *handle); +int libr_gtk_autoload(BuilderHandle **gtk_ret, IconList **icons_ret, int set_default_icon); +int libr_gtk_load(BuilderHandle **gtk_ret, char *resource_name); +int libr_glade_autoload(GladeHandle **glade_ret, IconList **icons_ret, int set_default_icon); +int libr_glade_load(GladeHandle **glade_ret, char *resource_name); + +#endif /* __LIBR_GTK_H */ + diff --git a/src/libr-i18n.c b/src/libr-i18n.c new file mode 100644 index 0000000..25e5664 --- /dev/null +++ b/src/libr-i18n.c @@ -0,0 +1,84 @@ +/* + * + *  Copyright (c) 2009 Erich Hoover + * + *  libr i18n - Add language resources into ELF binaries + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2.1 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * 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 + * + * To provide feedback, report bugs, or otherwise contact me: + * ehoover at mines dot edu + * + */ + +/* Include compile-time parameters */ +#include "config.h" + +#include "libr-i18n.h" +#include "tempfiles.h" + +/* Internationalization support */ +#include <locale.h> + +/* For string handling */ +#include <string.h> +#include <stdio.h> + +/* + * Extract the internationalization resources from the binary + * and setup gettext with the extracted folder. + */ +EXPORT_FN int libr_i18n_load(libr_file *handle, const char *domain) +{ +	char *temp_folder; +	int ret = true; +	 +	temp_folder = libr_extract_resources(handle); +	if(temp_folder == NULL) +		return false; +	if(!setlocale(LC_ALL, "")) +		ret = false; +	if(!bindtextdomain(domain, temp_folder)) +		ret = false; +	if(!textdomain(domain)) +		ret = false; +	if(!ret) +		cleanup_folder(temp_folder); +	else +		register_folder_cleanup(temp_folder); +	return ret; +} + +EXPORT_FN int libr_i18n_autoload(const char *domain) +{ +	libr_file *handle; +	 +	/* Obtain the handle to the executable */ +	if((handle = libr_open(NULL, LIBR_READ)) == NULL) +	{ +		/* "Failed to open this executable (%s) for resources", progname() */ +		return false; +	} +	/* Obtain the language files from the ELF binary */ +	if(!libr_i18n_load(handle, domain)) +	{ +		/* "Failed to load language resources!" */ +		goto failed; +	} +	 +failed: +	libr_close(handle); +	return true; +} diff --git a/src/libr-i18n.h b/src/libr-i18n.h new file mode 100644 index 0000000..5b61546 --- /dev/null +++ b/src/libr-i18n.h @@ -0,0 +1,14 @@ +#ifndef __LIBR_I18N_H +#define __LIBR_I18N_H + +#include "libr.h" +#include "gettext.h" + +#define _(string) gettext(string) +/* for strings used in structures (must manually call gettext!): */ +#define N_(string) (string) + +int libr_i18n_autoload(const char *domain); +int libr_i18n_load(libr_file *handle, const char *domain); + +#endif /* __LIBR_I18N_H */ diff --git a/src/libr-icons.c b/src/libr-icons.c new file mode 100644 index 0000000..18dc536 --- /dev/null +++ b/src/libr-icons.c @@ -0,0 +1,643 @@ +/* + * + *  Copyright (c) 2008-2011 Erich Hoover + * + *  libr icons - Add icon resources into ELF binaries + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2.1 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * 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 + * + * To provide feedback, report bugs, or otherwise contact me: + * ehoover at mines dot edu + * + */ + +/* Include compile-time parameters */ +#include "config.h" + +#include "libr-icons.h" + +/* For "one canvas" SVG documents */ +#include "onecanvas.h" + +/* For string manipulation */ +#include <stdlib.h> +#include <string.h> +#include <stdio.h> +#include <ctype.h> + +/* For handling files */ +#include <sys/stat.h> + +/* For C99 number types */ +#include <stdint.h> + +#define ICON_SECTION     ".icon" +#define TERM_LEN         1 + +#define OFFSET_ENTRIES   0 +#define OFFSET_GUID      OFFSET_ENTRIES+sizeof(uint32_t) + +#if defined(__i386) +	#define ID12FORMAT "%012llx" +#elif defined(__x86_64) +	#define ID12FORMAT "%012lx" +#else +	#define ID12FORMAT "%012lx" +	#warning "string formatting may be incorrect on this architecture." +#endif + +#ifndef DOXYGEN_SHOULD_SKIP_THIS + +typedef uint32_t ID8; +typedef uint16_t ID4; +typedef struct {uint64_t p:48;} __attribute__((__packed__)) ID12; + +typedef struct { +	ID8  g1; +	ID4  g2; +	ID4  g3; +	ID4  g4; +	ID12 g5; +} __attribute__((__packed__)) UUID; + +typedef struct { +	char *name; +	size_t offset; +	size_t entry_size; +	libr_icontype_t type; +	unsigned int icon_size; +} iconentry; + +typedef struct{ +	size_t size; +	char *buffer; +	iconentry entry; +} iconlist; + +#endif /* DOXYGEN_SHOULD_SKIP_THIS */ + +/* + * Decode a UUID to its binary representation + * + * NOTE: The last 12-bit parameter cannot be obtained using (uint64_t *) with + * some versions of GCC using some optimization levels.  This problem is very + * frustrating to debug, so I do not recommend playing with it yourself. + */ +UUID guid_decode(char *guid) +{ +	UUID id = {0x00000000, 0x0000, 0x0000, 0x0000, {0x000000000000} }; +	uint64_t tmp12; +	 +	sscanf(guid, "%08x-%04hx-%04hx-%04hx-" ID12FORMAT, &id.g1, &id.g2, &id.g3, &id.g4, &tmp12); +	id.g5.p = tmp12; +	return id; +} + +/* + * Return the size of the file represented by the file stream + */ +off_t fsize(FILE *handle) +{ +	struct stat file_stat; +	 +	if(fstat(fileno(handle), &file_stat) == ERROR) +		return ERROR; +	return file_stat.st_size; +} + +/* + * Create a new icon handle + */ +libr_icon *new_icon_handle(libr_icontype_t type, unsigned int icon_size, char *buffer, size_t buffer_size) +{ +	libr_icon *icon_handle = (libr_icon *) malloc(sizeof(libr_icon)); +	 +	icon_handle->type = type; +	icon_handle->buffer = buffer; +	icon_handle->icon_size = icon_size; +	icon_handle->buffer_size = buffer_size; +	return icon_handle; +} + +/* + * Obtain an existing icon resource list + */ +int get_iconlist(libr_file *file_handle, iconlist *icons) +{ +	if(icons == NULL) +	{ +		/* Need to be able to return SOMETHING */ +		return false; +	} +	/* Obtain the icon resource list */ +	icons->buffer = libr_malloc(file_handle, ICON_SECTION, &(icons->size)); +	if(icons->buffer == NULL) +		return false; +	return true; +} + +/* + * Get the next entry in an icon resource list + */ +iconentry *get_nexticon(iconlist *icons, iconentry *last_entry) +{ +	size_t i; +	 +	/* The icon list is needed both for the data buffer and for a call-specific iconentry instance */  +	if(icons == NULL) +		return NULL; +	/* If this is the first call (last_entry == NULL) then return the first entry */ +	if(last_entry == NULL) +		icons->entry.offset = sizeof(uint32_t)+sizeof(UUID); +	else +		icons->entry.offset += icons->entry.entry_size; +	/* Check to see if we've run out of entries */ +	if(icons->entry.offset >= icons->size) +		return NULL; +	i = icons->entry.offset; +	memcpy(&(icons->entry.entry_size), &(icons->buffer[i]), sizeof(uint32_t)); +	i += sizeof(uint32_t); +	icons->entry.type = icons->buffer[i]; +	i += sizeof(unsigned char); +	switch(icons->entry.type) +	{ +		case LIBR_SVG: +			icons->entry.icon_size = 0; +			icons->entry.name = &(icons->buffer[i]); +			break; +		case LIBR_PNG: +			memcpy(&(icons->entry.icon_size), &(icons->buffer[i]), sizeof(uint32_t)); +			i += sizeof(uint32_t); +			icons->entry.name = &(icons->buffer[i]); +			break; +		default: +			/* Invalid entry type */ +			return NULL; +	} +	return &(icons->entry); +} + +/* + * Free an icon handle + */ +EXPORT_FN int libr_icon_close(libr_icon *icon) +{ +	if(icon == NULL) +		return false; +	if(icon->buffer == NULL) +		return false; +	free(icon->buffer); +	free(icon); +	return true; +} + +/* + * Read an icon resource from an ELF file by name + */ +EXPORT_FN libr_icon *libr_icon_geticon_byname(libr_file *handle, char *icon_name) +{ +	iconentry *entry = NULL; +	libr_icon *icon = NULL; +	size_t buffer_size = 0; +	unsigned int icon_size; +	libr_icontype_t type; +	char *buffer = NULL; +	int inlist = false; +	iconlist icons; +	 +	if(!get_iconlist(handle, &icons)) +	{ +		/* Failed to obtain a list of ELF icons */ +		return NULL; +	} +	/* Look for the icon name in the entry list */ +	while((entry = get_nexticon(&icons, entry)) != NULL) +	{ +		if(!strcmp(entry->name, icon_name)) +		{ +			type = entry->type; +			icon_size = entry->icon_size; +			inlist = true; +			break; +		} +	} +	if(!inlist) +	{ +		/* Could not find icon name in the list of icons */ +		return false; +	} +	/* Get the icon from the ELF binary */ +	if(!libr_size(handle, icon_name, &buffer_size)) +	{ +		/* Failed to obtain ELF icon size */ +		return NULL; +	} +	/* Allocate memory for the icon */ +	buffer = (char *) malloc(buffer_size); +	if(buffer == NULL) +	{ +		/* Failed to allocate memory for icon */ +		return NULL; +	} +	/* Get the compressed icon from the ELF file */ +	if(!libr_read(handle, icon_name, buffer)) +	{ +		/* Failed to obtain ELF icon */ +		goto geticon_byname_complete; +	} +	icon = new_icon_handle(type, icon_size, buffer, buffer_size); +	 +geticon_byname_complete: +	if(icon == NULL) +		free(buffer); +	return icon; +} + +/* + * Read an icon resource from an ELF file by the square icon size + */ +EXPORT_FN libr_icon *libr_icon_geticon_bysize(libr_file *handle, unsigned int iconsize) +{ +	unsigned int closest_id = 0, i = 0, j = 0; +	int found_png = false, found_svg = false; +	unsigned long closest_size = 0; +	iconentry *entry = NULL; +	iconlist icons; +	 +	if(!get_iconlist(handle, &icons)) +	{ +		/* Failed to obtain a list of ELF icons */ +		return NULL; +	} +	/* Look for the closest size match, ignore SVG in case there are multiple icons */ +	while((entry = get_nexticon(&icons, entry)) != NULL) +	{ +		if(entry->type == LIBR_SVG) +			found_svg = true; +		if(entry->type == LIBR_PNG) +		{ +			if(j == 0) +			{ +				closest_size = entry->icon_size; +				found_png = true; +			} +			if(abs(iconsize-entry->icon_size) < closest_size) +			{ +				closest_size = entry->icon_size; +				closest_id = i; +			} +			j++; +		} +		i++; +	} +	/* If any PNG files were found then use the file if: +	 *  1) There are no SVG files <OR> +	 *  2) The PNG is an EXACT size match  +	 */ +	if(found_png) +	{ +		i=0; +		entry = NULL; +		while((entry = get_nexticon(&icons, entry)) != NULL) +		{ +			if(i == closest_id) +			{ +				if(entry->icon_size == iconsize || !found_svg) +					return libr_icon_geticon_byname(handle, entry->name); +				break; +			} +			i++; +		} +	} +	/* Otherwise use the SVG (provided that there is one) */ +	if(found_svg) +	{ +		entry = NULL; +		while((entry = get_nexticon(&icons, entry)) != NULL) +		{ +			if(entry->type == LIBR_SVG) +			{ +				libr_icon *icon = libr_icon_geticon_byname(handle, entry->name); +				if (icon) { +					libr_icon *icon_onecanvas; +					char *buffer; +				 +					/* should we report the requested size for SVG? */ +					icon->icon_size = iconsize; +				 +					/* if the SVG is a "one canvas" document then extract the correctly sized icon */ +					if((buffer = onecanvas_geticon_bysize(icon->buffer, iconsize)) != NULL) +					{ +						libr_icon_close(icon); +						icon_onecanvas = new_icon_handle(LIBR_SVG, iconsize, buffer, strlen(buffer)); +						return icon_onecanvas; +					} +				} +				return icon; +			} +		} +	} +	/* Give up */ +	return NULL; +} + +/* + * Obtains the icon UUID for the ELF file + */ +EXPORT_FN int libr_icon_getuuid(libr_file *handle, char *uuid) +{ +	UUID id = {0x00000000, 0x0000, 0x0000, 0x0000, {0x000000000000} }; +	iconlist icons; +	 +	if(!get_iconlist(handle, &icons)) +	{ +		/* Failed to obtain the list of ELF icons */ +		return false; +	} +	/* Now store the GUID to the return string */ +	memcpy(&id, &(icons.buffer[OFFSET_GUID]), sizeof(UUID)); +	snprintf(uuid, GUIDSTR_LENGTH, "%08x-%04hx-%04hx-%04hx-" ID12FORMAT "\n", id.g1, id.g2, id.g3, id.g4, (uint64_t) id.g5.p); +	free(icons.buffer); +	return true; +} +EXPORT_FN int libr_icon_getguid(libr_file *handle, char *uuid) ALIAS_FN(libr_icon_getuuid); + +/* + * Allocate a buffer containing the data of an icon + */ +EXPORT_FN char *libr_icon_malloc(libr_icon *icon, size_t *size) +{ +	char *iconfile = NULL; +	 +	if(size == NULL) +	{ +		/* No return size passed */ +		return NULL; +	} +	if(!libr_icon_size(icon, size)) +	{ +		/* Failed to obtain embedded icon file size */ +		return NULL; +	} +	iconfile = (char *) malloc(*size); +	if(!libr_icon_read(icon, iconfile)) +	{ +		/* Failed to obtain embedded icon file */ +		free(iconfile); +		return NULL; +	} +	return iconfile; +} + +/* + * Create an icon resource to represent a file on the hard disk  + */ +EXPORT_FN libr_icon *libr_icon_newicon_byfile(libr_icontype_t type, unsigned int icon_size, char *icon_file) +{ +	libr_icon *icon_handle = NULL; +	size_t len, buffer_size = 0; +	char *buffer = NULL; +	FILE *handle = NULL; +	 +	/* Open a handle to the icon file */ +	if((handle = fopen(icon_file, "r")) == NULL) +	{ +		/* Failed to open icon file */ +		return NULL; +	} +	/* Get the size of the icon file */ +	if((buffer_size = fsize(handle)) == ERROR) +	{ +		/* Failed to obtain the icon's file size */ +		return NULL; +	} +	/* Allocate a buffer for the uncompressed icon */ +	buffer = (char *) malloc(buffer_size); +	if(buffer == NULL) +	{ +		/* Failed to allocate a buffer for the icon data */ +		return NULL; +	} +	/* Read the uncompressed image from the disk */ +	if((len = fread(buffer, 1, buffer_size, handle)) <= 0) +	{ +		/* Failed to read icon from disk */ +		goto newicon_complete; +	} +	fclose(handle); +	if(len != buffer_size) +	{ +		/* Failed to read the entire icon */ +		goto newicon_complete; +	} +	/* Allocate the icon handle */ +	icon_handle = new_icon_handle(type, icon_size, buffer, buffer_size); +	 +newicon_complete: +	if(icon_handle == NULL) +		free(buffer); +	return icon_handle; +} + +/* + * Copy the icon resource into a buffer + */ +EXPORT_FN int libr_icon_read(libr_icon *icon, char *buffer) +{ +	if(icon == NULL) +		return false; +	memcpy(buffer, icon->buffer, icon->buffer_size); +	return true; +} + +/* + * Get the memory size of an icon resource + */ +EXPORT_FN int libr_icon_size(libr_icon *icon, size_t *size) +{ +	if(icon == NULL) +		return false; +	*size = icon->buffer_size; +	return true; +} + +/* + * Save the icon resource to a file + */ +EXPORT_FN int libr_icon_save(libr_icon *icon, char *filename) +{ +	FILE *file = NULL; +	int ret = false; +	size_t len; +	 +	if(icon == NULL) +		return false; +	/* Open the file to store the image */ +	if((file = fopen(filename, "w")) == NULL) +	{ +		/* Failed to open file to write the icon */ +		return false; +	} +	/* Store the uncompressed icon to disk */ +	if((len = fwrite(icon->buffer, 1, icon->buffer_size, file)) <= 0) +	{ +		/* Failed to write output file */ +		goto saveicon_complete; +	} +	if(len != icon->buffer_size) +	{ +		/* Did not write the entire file */  +		goto saveicon_complete; +	} +	ret = true; +	 +saveicon_complete: +	/* Close remaining resources */ +	fclose(file); +	return ret; +} + +/* + * Sets the icon GUID for the ELF file + */ +EXPORT_FN int libr_icon_setuuid(libr_file *handle, char *guid) +{ +	int ret = false; +	iconlist icons; +	UUID id; +	int i; +	 +	/* First check the GUID string */ +	for(i=0;i<strlen(guid);i++) +	{ +		if(!isxdigit(guid[i])) +		{ +			if(guid[i] == '-' && (i == 8 || i == 13 || i == 18 || i == 23)) +				continue; +			/* not a valid GUID string */ +			return false; +		} +	} +	id = guid_decode(guid); +	/* Now check existing resources */ +	if(!get_iconlist(handle, &icons)) +	{ +		/* No icons exist in the file, create a new icon section with the GUID */ +		uint32_t entries = 0; +		 +		icons.size = sizeof(uint32_t)+sizeof(UUID); +		icons.buffer = (char *) malloc(icons.size); +		memcpy(&(icons.buffer[OFFSET_ENTRIES]), &entries, sizeof(uint32_t)); +	} +	/* Set the GUID and write the resource */ +	if(!libr_write(handle, ICON_SECTION, icons.buffer, icons.size, LIBR_UNCOMPRESSED, LIBR_OVERWRITE)) +	{ +		/* failed to write icon resource */ +		goto setguid_complete; +	} +	ret = true; +	 +setguid_complete: +	free(icons.buffer); +	return ret; +} +EXPORT_FN int libr_icon_setguid(libr_file *handle, char *uuid) ALIAS_FN(libr_icon_setuuid); + +/* + * Add an icon resource to an ELF file + */ +EXPORT_FN int libr_icon_write(libr_file *handle, libr_icon *icon, char *icon_name, libr_overwrite_t overwrite) +{ +	size_t entry_size, i; +	iconentry *entry = NULL; +	iconlist icons; +	int ret = false; +	 +	/* Check to make sure the user did not make a poor name choice */ +	if(!strcmp(icon_name, ICON_SECTION)) +	{ +		/* ".icon" is a reserved section name */ +		return false; +	 +	} +	/* Check to make sure the file supports icon resources */ +	if(!get_iconlist(handle, &icons)) +	{ +		/* A GUID must be set first */ +		return false; +	} +	/* First add the icon as a new named section */ +	if(!libr_write(handle, icon_name, icon->buffer, icon->buffer_size, LIBR_COMPRESSED, overwrite)) +	{ +		/* Failed to add the icon as a resource */ +		goto writeicon_complete; +	} +	/* Look to see if the icon already has an entry */ +	while((entry = get_nexticon(&icons, entry)) != NULL) +	{ +		if(!strcmp(entry->name, icon_name)) +		{ +			ret = true; +			goto writeicon_complete; +		} +	} +	/* Now add the icon to the list of icon resources in the ".icon" section */ +	switch(icon->type) +	{ +		case LIBR_SVG: +			entry_size = sizeof(uint32_t)+sizeof(unsigned char)+strlen(icon_name)+TERM_LEN; +			break; +		case LIBR_PNG: +			entry_size = sizeof(uint32_t)+sizeof(unsigned char)+sizeof(uint32_t)+strlen(icon_name)+TERM_LEN; +			break; +		default: +			/* Unhandled icon type */ +			goto writeicon_complete; +	} +	icons.buffer = (char *) realloc(icons.buffer, icons.size+entry_size); +	if(icons.buffer == NULL) +	{ +		/* Failed to expand memory size */ +		goto writeicon_complete; +	} +	i = icons.size;  +	memcpy(&(icons.buffer[i]), &entry_size, sizeof(uint32_t)); +	i+=sizeof(uint32_t); +	icons.buffer[i] = icon->type; +	i+=sizeof(unsigned char); +	if(icon->type == LIBR_PNG) +	{ +		memcpy(&(icons.buffer[i]), &icon->icon_size, sizeof(uint32_t)); +		i+=sizeof(uint32_t); +	} +	memcpy(&(icons.buffer[i]), icon_name, strlen(icon_name)); +	i+=strlen(icon_name); +	icons.buffer[i] = '\0'; +	icons.size += entry_size; +	if(i != (icons.size-1)) +		printf("Really dangerous, buffer size mismatch!\n"); +	/* Write the updated icon table */ +	if(!libr_write(handle, ICON_SECTION, icons.buffer, icons.size, LIBR_UNCOMPRESSED, LIBR_OVERWRITE)) +	{ +		/* failed to write icon resource */ +		goto writeicon_complete; +	} +	ret = true; +	 +writeicon_complete: +	if(icons.buffer) +		free(icons.buffer); +	return ret; +} diff --git a/src/libr-icons.h b/src/libr-icons.h new file mode 100644 index 0000000..779fef3 --- /dev/null +++ b/src/libr-icons.h @@ -0,0 +1,201 @@ +/* + * + *  Copyright (c) 2008-2010 Erich Hoover + * + *  libr-icons - Handle icon resources in ELF binaries + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2.1 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * 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 + * + * To provide feedback, report bugs, or otherwise contact me: + * ehoover at mines dot edu + * + */ + +#ifndef __LIBR_ICONS_H +#define __LIBR_ICONS_H + +#include "libr.h" + +typedef enum { +	LIBR_SVG = 0, +	LIBR_PNG = 1 +} libr_icontype_t; + +#define UUIDSTR_LENGTH 37 +#define GUIDSTR_LENGTH UUIDSTR_LENGTH + +#ifdef __LIBR_BUILD__ +	typedef struct { +		char *buffer; +		size_t buffer_size; +		libr_icontype_t type; +		unsigned int icon_size; +	} libr_icon; +#else +	typedef void libr_icon; +#endif + +/************************************************************************* + * libr Icon API + *************************************************************************/ + +/** + * @page libr_icon_close Release an icon resource handle. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>int libr_icon_close(libr_icon *icon);</b> + *  + * @section DESCRIPTION + * 	Release the icon resource allocated by a call to + * 	<b>libr_icon_geticon_byid</b>(3), <b>libr_icon_geticon_byname</b>(3), + * 	<b>libr_icon_geticon_bysize</b>(3), <b>libr_icon_newicon_byfile</b>(3), + * 	or <b>libr_icon_newicon_frombuffer</b>(3). + * 	 + * 	@param icon The icon handle to release.  + * 	@return Returns 1 on success, 0 on failure. + *  + * @section SA SEE ALSO + * 	<b>libr_icon_geticon_byid</b>(3), <b>libr_icon_geticon_byname</b>(3), + * 	<b>libr_icon_geticon_bysize</b>(3), <b>libr_icon_newicon_byfile</b>(3), + * 	<b>libr_icon_newicon_frombuffer</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +int libr_icon_close(libr_icon *icon); + +/* +libr_icon *libr_icon_geticon_byid(libr_file *handle, unsigned int iconid); +*/ + +/** + * @page libr_icon_geticon_byname Retrieve an icon resource from an ELF + * 	binary (by the icon resource's name). + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>libr_icon *libr_icon_geticon_byname(libr_file *handle, char *iconname);</b> + *  + * @section DESCRIPTION + * 	Return a resource handle to an icon stored in a libr-compatible ELF + * 	binary.  When this handle is no-longer needed it must be unallocated + * 	using <b>libr_icon_close</b>(3).  + * 	 + * 	@param handle A handle returned by <b>libr_open</b>(3). + * 	@param iconname The exact name of the resource to return.  + * 	@return Returns a handle on success, NULL on failure. + *  + * @section SA SEE ALSO + * 	<b>libr_open</b>(3), <b>libr_icon_close</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +libr_icon *libr_icon_geticon_byname(libr_file *handle, char *iconname); + +/** + * @page libr_icon_geticon_bysize Retrieve an icon resource from an ELF + * 	binary (by the desired icon size). + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>libr_icon *libr_icon_geticon_bysize(libr_file *handle, unsigned int iconsize);</b> + *  + * @section DESCRIPTION + * 	Return a resource handle to the closest requested size icon stored + * 	in a libr-compatible ELF binary.  When this handle is no-longer + * 	needed it must be unallocated using <b>libr_icon_close</b>(3).  + * 	 + * 	@param handle A handle returned by <b>libr_open</b>(3). + * 	@param iconsize The size of the resource to return, use 0 + * 		to request an SVG icon. + * 	@return Returns a handle on success, NULL on failure. + *  + * @section SA SEE ALSO + * 	<b>libr_open</b>(3), <b>libr_icon_close</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +libr_icon *libr_icon_geticon_bysize(libr_file *handle, unsigned int iconsize); + +/** + * @page libr_icon_getuuid Retrieve the UUID of an application. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>int libr_icon_getuuid(libr_file *handle, char *uuid);</b> + *  + * @section DESCRIPTION + * 	Returns the icon UUID corresponding to the ELF binary in hex notation + * 	(XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX), which requires a 37 character + * 	buffer (36 data characters and a NULL terminator).  + * 	 + * 	@param handle A handle returned by <b>libr_open</b>(3). + * 	@param uuid A buffer to store the UUID of the application. + * 	@return Returns 1 on success, 0 on failure. + *  + * @section SA SEE ALSO + * 	<b>libr_open</b>(3), <b>libr_icon_close</b>(3), + * 		<b>libr_icon_setuuid</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +int libr_icon_getuuid(libr_file *handle, char *uuid); +DEPRECATED_FN int libr_icon_getguid(libr_file *handle, char *uuid); + +char *libr_icon_malloc(libr_icon *icon, size_t *size); +/* +libr_icon *libr_icon_newicon_frombuffer(libr_icontype_t type, int iconsize, char *buffer, size_t size); +*/ +libr_icon *libr_icon_newicon_byfile(libr_icontype_t type, unsigned int iconsize, char *iconfile); +/* +unsigned int libr_icon_num(libr_file *handle); +*/ +int libr_icon_read(libr_icon *icon, char *buffer); +int libr_icon_size(libr_icon *icon, size_t *size); +int libr_icon_save(libr_icon *icon, char *filename); + +/** + * @page libr_icon_setuuid Write a UUID into an application binary. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>int libr_icon_setuuid(libr_file *handle, char *uuid);</b> + *  + * @section DESCRIPTION + * 	Sets the icon UUID corresponding to the ELF binary in hex notation + * 	(XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX), which requires a 37 character + * 	buffer (36 data characters and a NULL terminator).  + * 	 + * 	@param handle A handle returned by <b>libr_open</b>(3). + * 	@param uuid A UUID to set for the application, can be generated by + * 		the terminal program "uuid". + * 	@return Returns 1 on success, 0 on failure. + *  + * @section SA SEE ALSO + * 	<b>libr_open</b>(3), <b>libr_icon_close</b>(3), + * 		<b>libr_icon_getuuid</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +int libr_icon_setuuid(libr_file *handle, char *uuid); +DEPRECATED_FN int libr_icon_setguid(libr_file *handle, char *guid); +int libr_icon_write(libr_file *handle, libr_icon *icon, char *iconname, libr_overwrite_t overwrite); + +#endif /* __LIBR_ICONS_H */ diff --git a/src/libr-internal.h b/src/libr-internal.h new file mode 100644 index 0000000..f7008a4 --- /dev/null +++ b/src/libr-internal.h @@ -0,0 +1,34 @@ +#ifndef __LIBR_INTERNAL_H +#define __LIBR_INTERNAL_H + +#define false                       0 +#define true                        1 +#define ERROR                      -1 +#define EXPORT_FN                  __attribute__((visibility ("protected"))) +#define INTERNAL_FN                __attribute__ ((visibility ("internal"))) +#define LIBR_TEMPFILE              "/tmp/libr-temp.XXXXXX" +#define LIBR_TEMPFILE_LEN          22 + +#ifndef DOXYGEN_SHOULD_SKIP_THIS + +typedef struct { +	char *message; +	libr_status status; +	const char *function; +} libr_intstatus; + +#endif /* DOXYGEN_SHOULD_SKIP_THIS */ + +struct _libr_file; + +void libr_set_error(libr_intstatus error); +libr_intstatus make_status(const char *function, libr_status code, char *message, ...); +/* Only called directly by cleanup routine, all other calls should be through libr_close */ +void libr_close_internal(struct _libr_file *file_handle); + +#define SET_ERROR(code,...)           make_status(__FUNCTION__, code, __VA_ARGS__) +#define RETURN(code,...)              return SET_ERROR(code, __VA_ARGS__) +#define RETURN_OK                     return SET_ERROR(LIBR_OK, NULL) +#define PUBLIC_RETURN(code,message)   {SET_ERROR(code, message); return (code == LIBR_OK);} + +#endif /* __LIBR_INTERNAL_H */ diff --git a/src/libr-link.h b/src/libr-link.h new file mode 100644 index 0000000..b1bdb54 --- /dev/null +++ b/src/libr-link.h @@ -0,0 +1,26 @@ +#ifndef __LIBR_LINK_H +#define __LIBR_LINK_H + +#ifndef DOXYGEN_SHOULD_SKIP_THIS + +typedef struct { +	void **symbol; +	char *name; +} symbol_table; + +#endif /* DOXYGEN_SHOULD_SKIP_THIS */ + +#define SYMBOL_TABLE(tbl, ...) \ +const symbol_table tbl[] = { \ +        __VA_ARGS__ \ +        {NULL, NULL} \ +} + +#define OVERRIDE_SYMBOL(a)            FN_##a +#define SYMBOL(sym)                   {(void **)&FN_##sym, #sym} +#define DEFINE_SYMBOL(ret, fn, ...)   ret (*OVERRIDE_SYMBOL(fn))(__VA_ARGS__) +#define LOAD_SYMBOLS(lib, tbl)        load_symbols(lib, tbl, sizeof(tbl)/sizeof(symbol_table)) + +int load_symbols(const char *library, const symbol_table *table, int entries); + +#endif /* __LIBR_LINK_H */ diff --git a/src/libr-ro.c b/src/libr-ro.c new file mode 100644 index 0000000..c3de28d --- /dev/null +++ b/src/libr-ro.c @@ -0,0 +1,351 @@ +/* + * + *  Copyright (c) 2009 Erich Hoover + *  Copyright (c) 2008-2009 Martin Rosenau + * + *  libr read-only Backend - Read resources from ELF binaries + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2.1 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * 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 + * + * To provide feedback, report bugs, or otherwise contact me: + * ehoover at mines dot edu + * + */ + +/* Include compile-time parameters */ +#include "config.h" + +#include "libr.h" +#include "libr-internal.h" + +/* malloc/free */ +#include <stdlib.h> + +/* For memory byte-wise compare */ +#include <string.h> + +/* For endian conversion */ +#include "cvtendian.h" + +#define RETURN_UNSUPPORTED RETURN(LIBR_ERROR_UNSUPPORTED, "The read-only backend does not support this operation"); + +#ifndef DOXYGEN_SHOULD_SKIP_THIS + +typedef struct { +	unsigned char magic[4]; +	eClass byte_size; +	eEncoding endian; +	unsigned char version; +	unsigned char padding[9]; +} ElfPreHeader; + +#endif /* DOXYGEN_SHOULD_SKIP_THIS */ + +#define ELF_HALF(b)  sizeof(uint16_t) +#define ELF_WORD(b)  sizeof(uint32_t) +#define ELF_XWORD(b) ((b == ELFCLASS32) ? sizeof(uint32_t) : sizeof(uint64_t)) +#define ELF_ADDR(b)  ELF_XWORD(b) +#define ELF_OFF(b)   ELF_XWORD(b) + +/* ELF Header Offsets */ +#define HDROFF_TYPE(b)      sizeof(ElfPreHeader)             /* ElfXX_Half  e_type; */ +#define HDROFF_MACHINE(b)   HDROFF_TYPE(b)+ELF_HALF(b)       /* ElfXX_Half  e_machine; */ +#define HDROFF_VERSION(b)   HDROFF_MACHINE(b)+ELF_HALF(b)    /* ElfXX_Word  e_version; */ +#define HDROFF_ENTRY(b)     HDROFF_VERSION(b)+ELF_WORD(b)    /* ElfXX_Addr  e_entry; */ +#define HDROFF_PHOFF(b)     HDROFF_ENTRY(b)+ELF_ADDR(b)      /* ElfXX_Off   e_phoff; */ +#define HDROFF_SHOFF(b)     HDROFF_PHOFF(b)+ELF_OFF(b)       /* ElfXX_Off   e_shoff; */ +#define HDROFF_FLAGS(b)     HDROFF_SHOFF(b)+ELF_OFF(b)       /* ElfXX_Word  e_flags; */ +#define HDROFF_EHSIZE(b)    HDROFF_FLAGS(b)+ELF_WORD(b)      /* ElfXX_Half  e_ehsize; */ +#define HDROFF_PHENTSIZE(b) HDROFF_EHSIZE(b)+ELF_HALF(b)     /* ElfXX_Half  e_phentsize; */ +#define HDROFF_PHNUM(b)     HDROFF_PHENTSIZE(b)+ELF_HALF(b)  /* ElfXX_Half  e_phnum; */ +#define HDROFF_SHENTSIZE(b) HDROFF_PHNUM(b)+ELF_HALF(b)      /* ElfXX_Half  e_shentsize; */ +#define HDROFF_SHNUM(b)     HDROFF_SHENTSIZE(b)+ELF_HALF(b)  /* ElfXX_Half  e_shnum; */ +#define HDROFF_SHSTRNDX(b)  HDROFF_SHNUM(b)+ELF_HALF(b)      /* ElfXX_Half  e_shstrndx; */ + +/* ELF Section Offsets */ +#define SECOFF_NAME(b)      0                                /* ElfXX_Word  sh_name; */ +#define SECOFF_TYPE(b)      SECOFF_NAME(b)+ELF_WORD(b)       /* ElfXX_Word  sh_type; */ +#define SECOFF_FLAGS(b)     SECOFF_TYPE(b)+ELF_WORD(b)       /* ElfXX_XWord sh_flags; */ +#define SECOFF_ADDR(b)      SECOFF_FLAGS(b)+ELF_XWORD(b)     /* ElfXX_Addr  sh_addr; */ +#define SECOFF_OFFSET(b)    SECOFF_ADDR(b)+ELF_ADDR(b)       /* ElfXX_Off   sh_offset; */ +#define SECOFF_SIZE(b)      SECOFF_OFFSET(b)+ELF_OFF(b)      /* ElfXX_XWord sh_size; */ +#define SECOFF_LINK(b)      SECOFF_SIZE(b)+ELF_XWORD(b)      /* ElfXX_Word  sh_link; */ +#define SECOFF_INFO(b)      SECOFF_LINK(b)+ELF_WORD(b)       /* ElfXX_Word  sh_info; */ +#define SECOFF_ADDRALIGN    SECOFF_INFO(b)+ELF_WORD(b)       /* ElfXX_XWord sh_addralign; */ +#define SECOFF_ENTSIZE      SECOFF_ADDRALIGN(b)+ELF_XWORD(b) /* ElfXX_XWord sh_entsize; */ + +/* + * Safely read a parameter from the ELF binary + */ +static int read_param(FILE *handle, void *result, size_t bytes, eClass endian) +{ +	if(fread(result, 1, bytes, handle) != bytes) +		return 0; +	if(ferror(handle)) +		return 0; +	if(endian != HOST_ENDIAN && !ConvertEndian(result, bytes)) +		return 0; +	return 1; +} + +/* + * The read-only backend requires no initialization + */ +void initialize_backend(void) +{ +	if(sizeof(ElfPreHeader) != 16) +		fprintf(stderr, "WARNING: Your compiler did not properly pack important structures!\n"); +} + +/* + * The read-only backend cannot write an output file + */ +void write_output(libr_file *file_handle) {} + +/* + * The read-only backend cannot add sections + */ +libr_intstatus add_section(libr_file *file_handle, char *resource_name, libr_section **retscn) +{ +	RETURN_UNSUPPORTED; +} + +/* + * Return the name of a section + */ +char *section_name(libr_file *file_handle, libr_section *scn) +{ +	if(scn == NULL) +		return NULL; +	return scn->name; +} + +/* + * Return the pointer to the actual data in the section + */ +void *data_pointer(libr_section *scn, libr_data *data) +{ +	return (void *) data; +} + +/* + * Return the size of the data in the section + */ +size_t data_size(libr_section *scn, libr_data *data) +{ +	return scn->size; +} + +/* + * Find the resource stored in the ELF binary + */ +libr_intstatus find_section(libr_file *file_handle, char *section, libr_section **retscn) +{ +	char *test_name; +	int i; +	 +	for(i=0; i<file_handle->total_sections; i++) +	{ +		test_name = section_name(file_handle, &(file_handle->secdata[i])); +		if(test_name != NULL && strcmp(test_name, section) == 0) +			break; +	} +	if(i >= file_handle->total_sections) +		RETURN(LIBR_ERROR_NOSECTION, "ELF resource section not found"); +	 +	/* Found the resource, hurray! */ +	*retscn = &(file_handle->secdata[i]); +	RETURN_OK; +} + +/* + * Read the section from the ELF binary + */ +libr_data *get_data(libr_file *file_handle, libr_section *scn) +{ +	FILE *handle = file_handle->handle; +	libr_data *data = NULL; +	size_t n; +	 +	fseek(handle, scn->data_offset, SEEK_SET); +	data = (libr_data *) malloc(scn->size); +	n = fread(data, 1, scn->size, handle); +	if(n == 0) +		goto failed; /* Empty section? */ +	if(ferror(handle)) +		goto failed; +	 +	/* Succeeded in reading the data */ +	return data; +failed: +	free(data); +	return NULL; +} + +/* + * UNSUPORTED BY BACKEND: Create a new data section + */ +libr_data *new_data(libr_file *file_handle, libr_section *scn) +{ +	return NULL; +} + +/* + * Find the next section given a section pointer + */ +libr_section *next_section(libr_file *file_handle, libr_section *scn) +{ +	int total_sections = file_handle->total_sections; +	libr_section *test_scn = NULL; +	int i; +	 +	if(total_sections == 0) +		return NULL; +	/* Requesting the first section */ +	if(scn == NULL) +	{ +		i = 0; +		/* Do not return an empty section */ +		while(test_scn == NULL || test_scn->size == 0) +		{ +			if(i > total_sections) +				return NULL; +			test_scn = &(file_handle->secdata[i++]); +		} +		return test_scn; +	} +	/* Return the next section given a section pointer */ +	for(i=0; i<total_sections; i++) +	{ +		test_scn = &(file_handle->secdata[i]); +		 +		if(test_scn == scn && (i+1) < total_sections) +		{ +			libr_section *next_scn = &(file_handle->secdata[i+1]); +			 +			/* Returning empty sections is pointless */ +			if(next_scn->size != 0) +				return next_scn; +		} +	} +	return NULL; +} + +/* + * UNSUPORTED BY BACKEND: Remove a section + */ +libr_intstatus remove_section(libr_file *file_handle, libr_section *scn) +{ +	RETURN_UNSUPPORTED; +} + +/* + * UNSUPORTED BY BACKEND: Set the data for a section + */ +libr_intstatus set_data(libr_file *file_handle, libr_section *scn, libr_data *data, off_t offset, char *buffer, size_t size) +{ +	RETURN_UNSUPPORTED; +} + +/* + * Open a handle to the ELF binary (provided that read-only access is requested) + */ +libr_intstatus open_handles(libr_file *file_handle, char *filename, libr_access_t access) +{ +	const char elf_magic[] = {'\x7F','E','L','F'}; +	uint16_t total_sections, sh_size, strings_sec; +	ElfPreHeader file_info; +	libr_section *secdata; +	FILE *handle = NULL; +	uint64_t sh_offset; +	unsigned long i; +	 +	if(access == LIBR_READ_WRITE) +		RETURN_UNSUPPORTED; +	handle = fopen(filename, "rb"); +	if(!handle) +		RETURN(LIBR_ERROR_OPENFAILED, "Failed to open input file"); +	if(fread(&file_info, 1, sizeof(ElfPreHeader), handle) != sizeof(ElfPreHeader)) +		RETURN(LIBR_ERROR_WRONGFORMAT, "Failed to read pre-header bytes from input file"); +	if(memcmp(file_info.magic, elf_magic, sizeof(elf_magic)) != 0) +		RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: not an ELF binary"); + +	/* Confirm processor (byte size) and packing (endian) */ +	if(!enum_valid(file_info.byte_size, ELFCLASS)) +		RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: invalid byte size"); +	if(!enum_valid(file_info.endian, ELFDATA)) +		RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: invalid endian type"); +	 +	/* Get the file offset to the Section Header tables */ +	fseek(handle, HDROFF_SHOFF(file_info.byte_size), SEEK_SET); +	if(!read_param(handle, &sh_offset, ELF_OFF(file_info.byte_size), file_info.endian)) +		RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: failed to read section header offset"); +	/* Get the size of the Section Header tables */ +	fseek(handle, HDROFF_SHENTSIZE(file_info.byte_size), SEEK_SET); +	if(!read_param(handle, &sh_size, ELF_HALF(file_info.byte_size), file_info.endian)) +		RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: failed to read section header size"); +	/* Get the total number of sections */ +	fseek(handle, HDROFF_SHNUM(file_info.byte_size), SEEK_SET); +	if(!read_param(handle, &total_sections, ELF_HALF(file_info.byte_size), file_info.endian)) +		RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: failed to read total number of sections"); +	/* Get the ID of the "strings" section */ +	fseek(handle, HDROFF_SHSTRNDX(file_info.byte_size), SEEK_SET); +	if(!read_param(handle, &strings_sec, ELF_HALF(file_info.byte_size), file_info.endian)) +		RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: failed to read string section ID"); +	if(strings_sec >= total_sections) +		RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: invalid string section ID"); +	secdata = (libr_section *) malloc(sizeof(libr_section)*total_sections); +	 +	/* Load section information */ +	for(i=0; i<total_sections; i++) +	{ +		long sec_start = sh_offset+sh_size*i; +		 +		/* Grab the offset in the string table to the name of the section */ +		fseek(handle, sec_start+SECOFF_NAME(file_info.byte_size), SEEK_SET); +		if(!read_param(handle, &(secdata[i].name_offset), ELF_WORD(file_info.byte_size), file_info.endian)) +			RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: failed to read section name offset"); +		/* Grab the offset to the data for the section */ +		fseek(handle, sec_start+SECOFF_OFFSET(file_info.byte_size), SEEK_SET); +		if(!read_param(handle, &(secdata[i].data_offset), ELF_OFF(file_info.byte_size), file_info.endian)) +			RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: failed to read section data offset"); +		/* Grab the size of the data for the section */ +		fseek(handle, sec_start+SECOFF_SIZE(file_info.byte_size), SEEK_SET); +		if(!read_param(handle, &(secdata[i].size), ELF_XWORD(file_info.byte_size), file_info.endian)) +			RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: failed to read section size"); +	} +	/* Locate the name offset within the "strings" section and load the string */ +	for(i=0; i<total_sections; i++) +	{ +		long stringsec_start = secdata[strings_sec].data_offset; +		size_t n; +		 +		fseek(handle, stringsec_start+secdata[i].name_offset, SEEK_SET); +		n = fread(secdata[i].name, 1, ELFSTRING_MAX-1, handle); +		if(ferror(handle)) +			RETURN(LIBR_ERROR_WRONGFORMAT, "Invalid input file format: failed to read string"); +		secdata[i].name[n] = '\0'; +	} +	 +	/* Hold onto the important parameters */ +	file_handle->secdata = secdata; +	file_handle->total_sections = total_sections; +	file_handle->endian = file_info.endian; +	file_handle->byte_size = file_info.byte_size; +	file_handle->handle = handle; +	file_handle->filename = filename; +	file_handle->access = access; +	RETURN_OK; +} diff --git a/src/libr-ro.h b/src/libr-ro.h new file mode 100644 index 0000000..8b8e41a --- /dev/null +++ b/src/libr-ro.h @@ -0,0 +1,62 @@ +#ifndef __LIBRRO_H +#define __LIBRRO_H + +/* For file handle support */ +#include <stdio.h> + +/* For integer types with set bit-sizes */ +#include <stdint.h> + +/*  + * NOTE: Packing the enum uses the smallest number of bytes + * possible to represent the value.  This packing does not + * guarantee that a "short enum" will be 8 bits, however, + * for the small enumerations in the ELF specification this + * IS the case (no enum requires more than 8 bits). + */ +#define SHORT_ENUM __attribute__ ((__packed__)) + +/* Type of byte-packing (endian) */ +typedef enum SHORT_ENUM { +	ELFDATANONE, /* Invalid */ +	ELFDATA2LSB, /* Least Significant Byte First */ +	ELFDATA2MSB, /* Most Significant Byte First */ +	ELFDATAMAX,  /* Invalid */ +} eEncoding; + +/* Type of target processor */ +typedef enum SHORT_ENUM { +	ELFCLASSNONE, /* Invalid */ +	ELFCLASS32,   /* 32-bit Field Alignment */ +	ELFCLASS64,   /* 64-bit Field Alignment */ +	ELFCLASSMAX,  /* Invalid */ +} eClass; + +#ifndef DOXYGEN_SHOULD_SKIP_THIS + +#define ELFSTRING_MAX 200 +typedef struct _libr_section { +	uint64_t size; +	uint64_t data_offset; +	uint32_t name_offset; +	char name[ELFSTRING_MAX]; +} libr_section; + +typedef struct _libr_file { +	FILE *handle; +	char *filename; +	eEncoding endian; +	eClass byte_size; +	libr_access_t access; +	libr_section *secdata; +	unsigned long total_sections; +} libr_file; + +#endif /* DOXYGEN_SHOULD_SKIP_THIS */ + +/* for a clean internal API */ +typedef void libr_data; + +#define enum_valid(val, name) (val > name##NONE && val < name##MAX) + +#endif /* __LIBRRO_H */ diff --git a/src/libr.c b/src/libr.c new file mode 100644 index 0000000..d038594 --- /dev/null +++ b/src/libr.c @@ -0,0 +1,489 @@ +/* + * + *  Copyright (c) 2008-2009 Erich Hoover + * + *  libr - Add resources into ELF binaries + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2.1 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * 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 + * + * To provide feedback, report bugs, or otherwise contact me: + * ehoover at mines dot edu + * + */ + +/* Include compile-time parameters */ +#include "config.h" + +#include "libr.h" +#include "tempfiles.h" + +/* Obtain file information */ +#include <sys/stat.h> +#include <stdlib.h> +#include <stdio.h> + +/* Compress files */ +#include <zlib.h> +#include <math.h> /* for ceil */ + +/* Handle strings and variable arguments*/ +#include <string.h> +#include <stdarg.h> + +/* For C99 number types */ +#include <stdint.h> + +/* Handle status codes for multiple threads */ +#include <pthread.h> + +#define SPEC_VERSION             '1' +#define OFFSET_TYPE              ((unsigned long) 4) +#define OFFSET_UNCOMPRESSED      ((unsigned long) OFFSET_TYPE+sizeof(unsigned char)) +#define OFFSET_UNCOMPRESSED_SIZE ((unsigned long) OFFSET_TYPE+sizeof(unsigned char)) +#define OFFSET_COMPRESSED        ((unsigned long) OFFSET_UNCOMPRESSED_SIZE+sizeof(uint32_t)) + +#if 0 + extern const char * __progname_full; + #define progpath() (char *) __progname_full +#endif +#define getself() ((char *) "/proc/self/exe") + +pthread_key_t error_key; + +/* + * Free the error status code/message structure + * (called on thread destruction or when a new code is set)  + */ +void free_error_key(void *_m) +{ +	libr_intstatus *error = (libr_intstatus *) _m; +	 +	if(error != NULL) +	{ +		/* Free the error structure */ +		if(error->message != NULL) +			free(error->message); +		free(error); +	} +} + +/* + * Set the error code and message for retrieval + */ +void libr_set_error(libr_intstatus error) +{ +	static int thread_key_initialized = false; +	libr_intstatus *status = NULL; +	 +	if(!thread_key_initialized) +	{ +		if(pthread_key_create(&error_key, free_error_key) != 0) +			return; /* a serious pthread-related error occurred */ +		if(pthread_setspecific(error_key, NULL) != 0) +			return; /* a serious pthread-related error occurred */ +		thread_key_initialized = true; +	} +	free_error_key(pthread_getspecific(error_key)); +	status = (libr_intstatus *) malloc(sizeof(libr_intstatus)); +	memcpy(status, &error, sizeof(libr_intstatus)); +	if(pthread_setspecific(error_key, (void *) status) != 0) +		return; /* a serious pthread-related error occurred */ +} + +/* + * Make an internal status passing structure, set the error code with this status + * if the status is not LIBR_OK. + */ +libr_intstatus make_status(const char *function, libr_status code, char *message, ...) +{ +	libr_intstatus status = {NULL, code, function}; +	va_list args; +	 +	if(message != NULL) +	{ +		status.message = (char *) malloc(1024); +		va_start(args, message); +		vsnprintf(status.message, 1024, message, args); +		va_end(args); +	} +	 +	libr_set_error(status); +	return status; +} + +/* + * Make sure that the section is libr-compatible + */ +libr_intstatus section_ok(libr_section *scn, libr_data *data) +{ +	char required_header[5], test_header[4] = {'R', 'E', 'S', SPEC_VERSION}; +	void *ptr = data_pointer(scn, data); +	size_t size = data_size(scn, data); +	 +	if(ptr == NULL || size < sizeof(required_header)) +		RETURN(LIBR_ERROR_NOTRESOURCE, "Not a valid libr-resource"); +	memcpy(required_header, ptr, sizeof(required_header)); +	if(strncmp(required_header, test_header, sizeof(test_header)) != 0) +		RETURN(LIBR_ERROR_NOTRESOURCE, "Not a valid libr-resource"); +	RETURN_OK; +} + +/* + * Remove a resourcefrom the ELF binary handle + */  +EXPORT_FN int libr_clear(libr_file *file_handle, char *resource_name) +{ +	libr_data *data = NULL; +	libr_section *scn = NULL; +	 +	/* Ensure valid inputs */ +	if(file_handle == NULL || resource_name == NULL) +		PUBLIC_RETURN(LIBR_ERROR_INVALIDPARAMS, "Invalid parameters passed to function"); +	if(file_handle->access != LIBR_READ_WRITE) +		PUBLIC_RETURN(LIBR_ERROR_NOPERM, "Open handle with LIBR_READ_WRITE access"); +	/* Find the section containing the icon */ +	if(find_section(file_handle, resource_name, &scn).status != LIBR_OK) +		return false; /* error already set */ +	/* Get the section data (interested in header) */ +	if((data = get_data(file_handle, scn)) == NULL) +		PUBLIC_RETURN(LIBR_ERROR_GETDATA, "Failed to obtain data of section"); +	/* Confirm that this resource is libr-compatible */ +	if(section_ok(scn, data).status != LIBR_OK) +		return false; /* error already set */ +	/* Clear the data resource */ +	if(set_data(file_handle, scn, data, 0, NULL, 0).status != LIBR_OK) +		return false; /* error already set */ +	/* Remove the section */ +	if(remove_section(file_handle, scn).status != LIBR_OK) +		return false; /* error already set */ +	return true; +} + +/* + * Close the specified ELF binary handle + */ +EXPORT_FN void libr_close(libr_file *file_handle) +{ +	unregister_handle_cleanup(file_handle); +	libr_close_internal(file_handle); +} +/* Only called directly by cleanup routine, all other calls should be through libr_close */ +void libr_close_internal(libr_file *file_handle) +{ +	write_output(file_handle); +	free(file_handle); +} + +/* + * Return the last error message for the active thread + */ +EXPORT_FN char *libr_errmsg(void) +{ +	libr_intstatus *error = (libr_intstatus *) pthread_getspecific(error_key); +	 +	if(error == NULL) +		return NULL; +	return error->message; +} + +/* + * Return the last error code for the active thread (or LIBR_OK for no error) + */ +EXPORT_FN libr_status libr_errno(void) +{ +	libr_intstatus *error = (libr_intstatus *) pthread_getspecific(error_key); +	 +	if(error == NULL) /* Nothing has happened yet */ +		return LIBR_OK; +	return error->status; +} + +/* + * Return the name of a libr-compatible resource + */ +EXPORT_FN char *libr_list(libr_file *file_handle, unsigned int resourceid) +{ +	libr_section *scn = NULL; +	libr_data *data = NULL; +	int i = 0; +	 +	while((scn = next_section(file_handle, scn)) != NULL) +	{ +		/* Get the section data (interested in header) */ +		if((data = get_data(file_handle, scn)) == NULL) +			return NULL; +		if(section_ok(scn, data).status == LIBR_OK) +		{ +			if(i == resourceid) +				return strdup(section_name(file_handle, scn)); +			i++; +		} +	} +	return NULL; +} + +/* + * Allocate a buffer containing the data of a resource + */ +EXPORT_FN char *libr_malloc(libr_file *file_handle, char *resource_name, size_t *size) +{ +	char *buffer = NULL; +	size_t size_local; +	 +	if(size == NULL) +		size = &size_local; +	if(!libr_size(file_handle, resource_name, size)) +		return NULL; /* error already set */ +	buffer = (char *) malloc(*size); +	if(!libr_read(file_handle, resource_name, buffer)) +	{ +		free(buffer); +		return NULL; /* error already set */  +	} +	return buffer; +} + +/* + * Open the specified ELF binary (caller if filename is NULL) + */ +EXPORT_FN libr_file *libr_open(char *filename, libr_access_t access) +{ +	libr_file *file_handle = NULL; +	static int initialized = false; +	 +	if(!initialized) +	{ +		if(strncmp(zlibVersion(), ZLIB_VERSION, 1) != 0) +		{ +			SET_ERROR(LIBR_ERROR_ZLIBINIT, "zlib library initialization failed");  +			return NULL; +		} +		initialize_backend(); +		initialized = true; +	} +	 +	if(filename == NULL) +		filename = getself(); +	file_handle = (libr_file *) malloc(sizeof(libr_file)); +	memset(file_handle, 0, sizeof(libr_file)); +	if(open_handles(file_handle, filename, access).status != LIBR_OK) +	{ +		/* failed to open file for processing, error already set */  +		free(file_handle); +		file_handle = NULL; +	} +	/* Cleanup handles automatically when libr exits memory */ +	if(file_handle != NULL) +		register_handle_cleanup(file_handle); +	return file_handle; +} + +/* + * Read a resource from the specified ELF binary handle + */ +EXPORT_FN int libr_read(libr_file *file_handle, char *resource_name, char *buffer) +{ +	unsigned long uncompressed_size = 0, compressed_size = 0; +	char *data_buffer = NULL; +	libr_section *scn = NULL; +	libr_data *data = NULL; +	libr_type_t type; +	 +	/* Find the section containing the icon */ +	if(find_section(file_handle, resource_name, &scn).status != LIBR_OK) +		return false; /* error already set */ +	/* Get the section data (interested in header) */ +	if((data = get_data(file_handle, scn)) == NULL) +		PUBLIC_RETURN(LIBR_ERROR_GETDATA, "Failed to obtain data of section"); +	/* Confirm that this resource is libr-compatible */ +	if(section_ok(scn, data).status != LIBR_OK) +		return false; /* error already set */ +	data_buffer = (char *) data_pointer(scn, data); +	/* Get the size of the data resource */ +	type = (libr_type_t) data_buffer[OFFSET_TYPE]; +	switch(type) +	{ +		case LIBR_UNCOMPRESSED: +		{	if(data_size(scn, data)-OFFSET_UNCOMPRESSED < 0) +				PUBLIC_RETURN(LIBR_ERROR_SIZEMISMATCH, "Section's data size does not make sense"); +			uncompressed_size = data_size(scn, data)-OFFSET_UNCOMPRESSED; +			memcpy(buffer, &data_buffer[OFFSET_UNCOMPRESSED], uncompressed_size); +		}	break; +		case LIBR_COMPRESSED: +		{ +			uint32_t size_temp; +			 +			memcpy(&size_temp, &data_buffer[OFFSET_UNCOMPRESSED_SIZE], sizeof(uint32_t)); +			uncompressed_size = size_temp; +			compressed_size = data_size(scn, data)-OFFSET_COMPRESSED; +			if(uncompress((unsigned char *)buffer, &uncompressed_size, (unsigned char *)&data_buffer[OFFSET_COMPRESSED], compressed_size) != Z_OK) +				PUBLIC_RETURN(LIBR_ERROR_UNCOMPRESS, "Failed to uncompress resource data"); +		}	break; +		default: +			PUBLIC_RETURN(LIBR_ERROR_INVALIDTYPE, "Invalid data storage type specified"); +	} +	return true; +} + +/* + * Retrieve the number of libr-compatible resources + */ +EXPORT_FN unsigned int libr_resources(libr_file *file_handle) +{ +	libr_section *scn = NULL; +	libr_data *data = NULL; +	int i = 0; +	 +	while((scn = next_section(file_handle, scn)) != NULL) +	{ +		if((data = get_data(file_handle, scn)) == NULL) +			continue; +		if(section_ok(scn, data).status == LIBR_OK) +			i++; +	} +	return i; +} + +/* + * Get the size of a resource from the specified ELF binary handle + */ +EXPORT_FN int libr_size(libr_file *file_handle, char *resource_name, size_t *retsize) +{ +	char *data_buffer = NULL; +	libr_section *scn = NULL; +	libr_data *data = NULL; +	unsigned long size = 0; +	libr_type_t type; +	 +	/* Find the section containing the icon */ +	if(find_section(file_handle, resource_name, &scn).status != LIBR_OK) +		return false; /* error already set */ +	/* Get the section data (interested in header) */ +	if((data = get_data(file_handle, scn)) == NULL) +		PUBLIC_RETURN(LIBR_ERROR_GETDATA, "Failed to obtain data of section"); +	/* Confirm that this resource is libr-compatible */ +	if(section_ok(scn, data).status != LIBR_OK) +		return false; /* error already set */ +	data_buffer = (char *) data_pointer(scn, data); +	/* Get the size of the data resource */ +	type = (libr_type_t) data_buffer[OFFSET_TYPE]; +	switch(type) +	{ +		case LIBR_UNCOMPRESSED: +		{ +			size_t full_size = data_size(scn, data); +			  +			if(full_size-OFFSET_UNCOMPRESSED < 0) +				PUBLIC_RETURN(LIBR_ERROR_SIZEMISMATCH, "Section's data size does not make sense");  +			size = full_size - OFFSET_UNCOMPRESSED; +		}	break; +		case LIBR_COMPRESSED: +		{ +			memcpy(&size, &data_buffer[OFFSET_UNCOMPRESSED_SIZE], sizeof(uint32_t)); +		}	break; +		default: +			PUBLIC_RETURN(LIBR_ERROR_INVALIDTYPE, "Invalid data storage type specified"); +	} +	*retsize = size; +	return true; +} + +/* + * Write a resource to the specified ELF binary handle + */ +EXPORT_FN int libr_write(libr_file *file_handle, char *resource_name, char *buffer, size_t size, libr_type_t type, libr_overwrite_t overwrite) +{ +	char header[9] = {'R', 'E', 'S', SPEC_VERSION}; +	unsigned int header_size = 4; +	libr_section *scn = NULL; +	libr_data *data = NULL; +	libr_intstatus ret; +	 +	/* Ensure valid inputs */ +	if(file_handle == NULL || resource_name == NULL || buffer == NULL) +		PUBLIC_RETURN(LIBR_ERROR_INVALIDPARAMS, "Invalid parameters passed to function"); +	if(file_handle->access != LIBR_READ_WRITE) +		PUBLIC_RETURN(LIBR_ERROR_NOPERM, "Open handle with LIBR_READ_WRITE access"); +	/* Get the section if it already exists */ +	ret = find_section(file_handle, resource_name, &scn); +	if(ret.status == LIBR_OK) +	{ +		/* If the section exists (and overwrite is not specified) then fail */ +		if(!overwrite) +			PUBLIC_RETURN(LIBR_ERROR_OVERWRITE, "Section already exists, over-write not specified");  +		/* Grab the existing data section for overwriting */ +		if((data = get_data(file_handle, scn)) == NULL) +			PUBLIC_RETURN(LIBR_ERROR_GETDATA, "Failed to obtain data of section"); +	} +	else if(ret.status == LIBR_ERROR_NOSECTION) +	{ +		/* Create a new section named "resource_name" */ +		if(add_section(file_handle, resource_name, &scn).status != LIBR_OK) +			return false; /* error already set */ +		/* Create a data segment to store the compressed image */ +		if((data = new_data(file_handle, scn)) == NULL) +			PUBLIC_RETURN(LIBR_ERROR_NEWDATA, "Failed to create data for section"); +	} +	else +		return false; /* error already set */ +	 +	header[header_size++] = (char) type; +	switch(type) +	{ +		case LIBR_UNCOMPRESSED: +			/* Do nothing, just stick the data in */ +			break; +		case LIBR_COMPRESSED: +		{ +			char *compressed_buffer = NULL, *uncompressed_buffer = buffer; +			unsigned long compressed_size = 0, uncompressed_size = size; +			uint32_t size_temp; +			  +			/* Store the uncompressed size to the header */ +			size_temp = uncompressed_size; +			memcpy(&header[header_size], &size_temp, sizeof(uint32_t)); +			header_size += sizeof(uint32_t); +			/* Compress the data for storage */ +			compressed_size = ceil((uncompressed_size+12)*1.1); +			compressed_buffer = (char *) malloc(compressed_size); +			if(compress((unsigned char *)compressed_buffer, &compressed_size, (unsigned char *)uncompressed_buffer, uncompressed_size) != Z_OK) +			{ +				free(compressed_buffer); +				PUBLIC_RETURN(LIBR_ERROR_COMPRESS, "Failed to compress resource data"); +			} +			/* From here on treat the compressed buffer as the data */ +			buffer = compressed_buffer; +			size = compressed_size; +		}	break; +		default: +			PUBLIC_RETURN(LIBR_ERROR_INVALIDTYPE, "Invalid data storage type specified"); +	} +	/* Store the resource header data */ +	if(set_data(file_handle, scn, data, 0, &header[0], header_size).status != LIBR_OK) +		return false; /* error already set */ +	/* Create a data segment to store the post-header data +	 * NOTE: For existing files the data of the section is represented as a continuous stream +	 * (so calling elf_getdata now WILL NOT return the post-header data) +	 */ +	if((data = new_data(file_handle, scn)) == NULL) +		PUBLIC_RETURN(LIBR_ERROR_NEWDATA, "Failed to create data for section"); +	/* Store the actual user data to the section */ +	if(set_data(file_handle, scn, data, header_size, buffer, size).status != LIBR_OK) +		return false; /* error already set */ +	/* Close compression resources */ +	if(type == LIBR_COMPRESSED) +		free(buffer); +	return true; +} diff --git a/src/libr.h b/src/libr.h new file mode 100644 index 0000000..b1aa1d7 --- /dev/null +++ b/src/libr.h @@ -0,0 +1,416 @@ +/* + * + *  Copyright (c) 2008-2011 Erich Hoover + * + *  libr - Add resources into ELF binaries + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2.1 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * 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 + * + * To provide feedback, report bugs, or otherwise contact me: + * ehoover at mines dot edu + * + */ + +#ifndef __LIBR_H +#define __LIBR_H + +#include <sys/types.h> + +#define DEPRECATED_FN           __attribute__ ((deprecated)) +#define ALIAS_FN(fn)            __attribute__ ((weak, alias (#fn))) + +/** + * @addtogroup libr_status libr_status + * @brief Enumeration of possible libr status values. + * @{ + * \#include <libr.h> + */ +/** Possible libr status values */ +typedef enum { +	LIBR_OK                     =   0, /**< Success */ +	LIBR_ERROR_GETEHDR          =  -1, /**< Failed to obtain ELF header: */ +	LIBR_ERROR_NOTABLE          =  -2, /**< No ELF string table */ +	LIBR_ERROR_TABLE            =  -3, /**< Failed to open string table: */ +	LIBR_ERROR_GETDATA          =  -4, /**< Failed to obtain data of section */ +	LIBR_ERROR_GETSHDR          =  -5, /**< Failed to obtain ELF section header: */ +	LIBR_ERROR_SIZEMISMATCH     =  -6, /**< Section's data size does not make sense */ +	LIBR_ERROR_UPDATE           =  -7, /**< Failed to perform dynamic update: */ +	LIBR_ERROR_NEWSECTION       =  -8, /**< Failed to create new section */ +	LIBR_ERROR_NEWDATA          =  -9, /**< Failed to create data for section */ +	LIBR_ERROR_REMOVESECTION    = -10, /**< Failed to remove section: */ +	LIBR_ERROR_NOSECTION        = -11, /**< ELF resource section not found */ +	LIBR_ERROR_STRPTR           = -12, /**< Failed to obtain section string pointer: */ +	LIBR_ERROR_NOTRESOURCE      = -13, /**< Not a valid libr-resource */ +	LIBR_ERROR_EXPANDSECTION    = -14, /**< Failed to expand section */ +	LIBR_ERROR_WRONGFORMAT      = -15, /**< Invalid input file format */ +	LIBR_ERROR_SETFLAGS         = -16, /**< Failed to set flags for section */ +	LIBR_ERROR_NOPERM           = -17, /**< Open handle with LIBR_READ_WRITE access */ +	LIBR_ERROR_NOSIZE           = -18, /**< Failed to obtain file size */ +	LIBR_ERROR_SETFORMAT        = -19, /**< Failed to set output file format to input file format */ +	LIBR_ERROR_SETARCH          = -20, /**< Failed to set output file architecture to input file architecture */ +	LIBR_ERROR_OVERWRITE        = -21, /**< Section already exists, over-write not specified */ +	LIBR_ERROR_COMPRESS         = -22, /**< Failed to compress resource data */ +	LIBR_ERROR_INVALIDTYPE      = -23, /**< Invalid data storage type specified */ +	LIBR_ERROR_MEMALLOC         = -24, /**< Failed to allocate memory for data */ +	LIBR_ERROR_INVALIDPARAMS    = -25, /**< Invalid parameters passed to function */ +	LIBR_ERROR_UNCOMPRESS       = -26, /**< Failed to uncompress resource data */ +	LIBR_ERROR_ZLIBINIT         = -27, /**< zlib library initialization failed */ +	LIBR_ERROR_OPENFAILED       = -28, /**< Failed to open input file */ +	LIBR_ERROR_BEGINFAILED      = -29, /**< Failed to open ELF file: */ +	LIBR_ERROR_WRITEPERM        = -30, /**< No write permission for file */ +	LIBR_ERROR_UNSUPPORTED      = -31, /**< The requested operation is not supported by the backend */ +} libr_status; +/** + * @} + */ + +typedef enum { +	LIBR_READ       = 0, +	LIBR_READ_WRITE = 1, +} libr_access_t; + +typedef enum { +	LIBR_UNCOMPRESSED = 0, +	LIBR_COMPRESSED   = 1 +} libr_type_t; + +typedef enum { +	LIBR_NOOVERWRITE = 0, +	LIBR_OVERWRITE   = 1 +} libr_overwrite_t; + +#ifdef __LIBR_BUILD__ +	#include "libr-internal.h" +	#if __LIBR_BACKEND_libbfd__ +		#include "libr-bfd.h" +	#elif __LIBR_BACKEND_libelf__ +		#include "libr-elf.h" +	#elif __LIBR_BACKEND_readonly__ +		#include "libr-ro.h" +	#else /* LIBR_BACKEND */ +		#error "Unhandled backend" +	#endif /* LIBR_BACKEND */ +	#include "libr-backends.h" +#else +	struct _libr_file; +	typedef struct _libr_file libr_file; +#endif /* __LIBR_BUILD__ */ + +/************************************************************************* + * libr Resource Management API + *************************************************************************/ + +/** + * @page libr_clear Remove a resource from an ELF executable. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>int libr_clear(libr_file *handle, char *resourcename);</b> + *  + * @section DESCRIPTION + * 	Removes a libr-compatible resource from an ELF executable.  The handle + * 	must be opened using <b>libr_open</b>(3) with either <b>LIBR_WRITE</b> + * 	 or <b>LIBR_READ_WRITE</b> access in order to remove a resource. + *  + * 	Please note that resource removal does not occur until the handle is + * 	closed using <b>libr_close</b>(3). + * 	 + * 	@param handle A handle returned by <b>libr_open</b>(3). + * 	@param resourcename The name of the libr-compatible resource to remove.  + *  + * @section SA SEE ALSO + * 	<b>libr_open</b>(3), <b>libr_close</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +int libr_clear(libr_file *handle, char *resourcename); + +/** + * @page libr_close Close a handle to an ELF executable. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>void libr_close(libr_file *handle);</b> + *  + * @section DESCRIPTION + * 	Handles opened with <b>libr_open</b>(3) should be closed with + * 	<b>libr_close</b>() when they are no-longer needed by the calling + * 	application.  + * 	 + * 	@param handle The handle to close.  + *  + * @section SA SEE ALSO + * 	<b>libr_open</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +void libr_close(libr_file *handle); + +/** + * @page libr_errmsg Return a detailed description of the last + * 	libr-related error. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>char *libr_errmsg(void);</b> + *  + * @section DESCRIPTION + * 	Returns a detailed string describing the last error encountered by + * 	the libr resource library.  The string is an internal error + * 	description, so it should not be freed. + * 	 + * 	If no errors have been encountered then NULL is returned.  + *  + * @section SA SEE ALSO + * 	<b>libr_errno</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +char *libr_errmsg(void); + +/** + * @page libr_errno Return a status code describing the last + * 	libr-related error. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>libr_status libr_errno(void);</b> + *  + * @section DESCRIPTION + * 	Returns a code corresponding to the last error encountered by + * 	the libr resource library.  For a detailed description of possible + * 	return values see <b>libr_status</b>(3). + * 	 + * 	To get a user-readable string corresponding to the last error the + * 	<b>libr_errmsg</b>(3) function should be used instead.  + * 	 + * 	If no errors have been encountered then <b>LIBR_OK</b> is returned.  + *  + * @section SA SEE ALSO + * 	<b>libr_errmsg</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +libr_status libr_errno(void); + +/** + * @page libr_list Obtain the name of a libr ELF resource (by index). + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>char *libr_list(libr_file *file_handle, unsigned int resourceid);</b> + *  + * @section DESCRIPTION + * 	Returns the name of a libr-compatible resource stored in an ELF binary + * 	corresponding to the given resource index.  The index value ranges from + * 	0 to the value returned by <b>libr_resources</b>(3), which returns the + * 	total number of libr-compatible resources stored in the ELF binary. + * 	 + * 	@param handle A handle returned by <b>libr_open</b>(3). + * 	@param resourceid The index of the libr-compatible resource for which + * 		the name will be returned. + * 	 + * 	@return Returns a string containing the name of the resource section.  This + * 		string is allocated when the function is called, so it <i>must be + * 		unallocated</i> with a call to <b>free</b>(3) when it is no-longer + * 		needed.  NULL is returned on failure. + *  + * @section SA SEE ALSO + * 	<b>libr_open</b>(3), <b>free</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +char *libr_list(libr_file *file_handle, unsigned int resourceid); + +/** + * @page libr_malloc Obtain the data corresponding to a libr ELF resource. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>char *libr_malloc(libr_file *handle, char *resourcename, size_t *size);</b> + *  + * @section DESCRIPTION + * 	Returns the contents of a libr-compatible resource stored in an ELF binary + * 	corresponding to the given resource name. + * 	 + * 	@param handle A handle returned by <b>libr_open</b>(3). + * 	@param resourcename The name of the libr-compatible resource for which + * 		the data will be returned. + * 	@param size A pointer for storing the length of the data contained in the + * 		the resource.  May be NULL. + * 	 + * 	@return Returns NULL on failure, the pointer to a buffer containing the data + * 		for the resource on success.  When the buffer is no-longer used it must + * 		be unallocated using a call to <b>free</b>(3).  + *  + * @section SA SEE ALSO + * 	<b>libr_open</b>(3), <b>free</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +char *libr_malloc(libr_file *handle, char *resourcename, size_t *size); + +/** + * @page libr_open Open an ELF executable file for resource management. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>libr_file *libr_open(char *filename, libr_access_t access);</b> + *  + * @section DESCRIPTION + * 	<b>libr_open</b>() can be used on any ELF executable, however, + * 	<b>libr_open</b>() called with <b>LIBR_READ</b> access is only useful + * 	for executables that already contain libr-compatible stored resources. + * 	 + * 	An application can easily access its own resources by passing NULL for + * 	the filename and requesting <b>LIBR_READ</b> access.  For the obvious + * 	reason that an actively-open application cannot edit itself, the + * 	calling binary may only request <b>LIBR_READ</b> access. + * 	 + * 	@param filename ELF executable to manage.  Pass a NULL pointer as the + * 		filename in order to access the calling binary (<b>LIBR_READ</b> + * 		access only) @param access Requested access type (<b>LIBR_READ</b>, + * 		<b>LIBR_WRITE</b>, <b>LIBR_READ_WRITE</b>), the valid operations for + * 		the returned handle will be restricted based upon the requested access. + * 	@return Returns a libr file handle on success, NULL on failure.  The + * 		handle should be freed with <b>libr_close</b>(3) when no-longer used.  + *  + * @section SA SEE ALSO + * 	<b>libr_close</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +libr_file *libr_open(char *filename, libr_access_t access); + +/** + * @page libr_read Read out the contents of a libr ELF resource. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>int libr_read(libr_file *handle, char *resourcename, char *buffer);</b> + * + * @section WARNING + * 	This function does not allocate memory for the buffer, so the buffer must + * 	be large enough to fit the resource data.  For this reason it is suggested + * 	that <b>libr_malloc</b>(3) be used in preference over this function. + *  + * @section DESCRIPTION + * 	Reads the contents of a resource embedded in an ELF binary, the resource + * 	must be compatible with the libr specification. + * 	 + * 	@param handle A handle returned by <b>libr_open</b>(3). + * 	@return Returns 1 on success, 0 on failure.  + *  + * @section SA SEE ALSO + * 	<b>libr_open</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +int libr_read(libr_file *handle, char *resourcename, char *buffer); + +/** + * @page libr_resources Returns the number of resources contained in + * 	the ELF binary. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>unsigned int libr_resources(libr_file *handle);</b> + * + * @section DESCRIPTION + * 	Returns the total number of libr-compatible resources contained + * 	in the ELF binary.  Intended to be used with <b>libr_list</b>(3) + * 	to return the full list of resources contained in the binary.  + * 	 + * 	@param handle A handle returned by <b>libr_open</b>(3). + * 	@return The total number of libr resources in the binary. + *  + * @section SA SEE ALSO + * 	<b>libr_open</b>(3), <b>libr_list</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +unsigned int libr_resources(libr_file *handle); + +/** + * @page libr_size Returns the uncompressed size of a libr resource. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>int libr_size(libr_file *handle, char *resourcename, size_t *size);</b> + * + * @section DESCRIPTION + * 	Obtain the total number of bytes consumed by the uncompressed + * 	version of the specific libr-resource.  Intended to be used with + * 	<b>libr_read</b>(3) in order to allocate a large enough buffer + * 	for the resource. + * 	 + * 	@param handle A handle returned by <b>libr_open</b>(3). + * 	@param resourcename The name of the resource for which the + * 		size of the data section will be returned. + * 	@param size A pointer for storing the size of the data section. + * 		This pointer cannot be NULL. + * 	@return Returns 1 on success, 0 on failure.  + *  + * @section SA SEE ALSO + * 	<b>libr_open</b>(3), <b>libr_read</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +int libr_size(libr_file *handle, char *resourcename, size_t *size); + +/** + * @page libr_write Adds a libr resource to an ELF binary. + * @section SYNOPSIS + * 	\#include <libr.h> + * 	 + * 	<b>int libr_write(libr_file *handle, char *resourcename, char *buffer, size_t size, libr_type_t type, libr_overwrite_t overwrite);</b> + * + * @section DESCRIPTION + * 	Adds a libr-compatible resource into the ELF binary.  The handle + * 	must be opened using <b>libr_open</b>(3) with either <b>LIBR_WRITE</b> + * 	or <b>LIBR_READ_WRITE</b> access in order to add a resource. + * 	 + * 	@param handle A handle returned by <b>libr_open</b>(3). + * 	@param resourcename The name of the resource to create. + * 	@param buffer A string containing the data of the resource. + * 	@param size The total size of the buffer. + * 	@param type The method which should be used for storing the  + * 		data (either <b>LIBR_UNCOMPRESSED</b> or + * 		<b>LIBR_COMPRESSED</b>). + * 	@param overwrite Whether overwriting an existing resource + * 		should be permitted (either <b>LIBR_NOOVERWRITE</b> or + * 		<b>LIBR_OVERWRITE</b>).  + * 	@return Returns 1 on success, 0 on failure.  + *  + * @section SA SEE ALSO + * 	<b>libr_open</b>(3) + *  + * @section AUTHOR + * 	Erich Hoover <ehoover@mines.edu> + */ +int libr_write(libr_file *handle, char *resourcename, char *buffer, size_t size, libr_type_t type, libr_overwrite_t overwrite); + +#endif /* __LIBR_H */ + diff --git a/src/onecanvas.c b/src/onecanvas.c new file mode 100644 index 0000000..e53ece7 --- /dev/null +++ b/src/onecanvas.c @@ -0,0 +1,446 @@ +/* + * + *  Copyright (c) 2010 Erich Hoover + * + *  libr "one canvas" - Handle multiple icons stored in a single "one canvas"  + *                      SVG document. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2.1 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * 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 + * + * To provide feedback, report bugs, or otherwise contact me: + * ehoover at mines dot edu + * + */ +  +#include <stdio.h> +#include <string.h> +#include <math.h> +#include <stdlib.h> +#include <time.h> + +#define FALSE 0 +#define TRUE  1 + +typedef struct { +	double x; +	double y; +	double width; +	double height; +	int icon_width; +	int icon_height; +} IconSVG; + +typedef enum { +	STATUS_FINDSVG, +	STATUS_FINDMETADATA, +	STATUS_FINDPUBLISHER_START, +	STATUS_FINDPUBLISHER_STOP, +	STATUS_FINDHIDDEN, +	STATUS_FINDBOUNDS, +	STATUS_FAILED, +	STATUS_DONE, +} eStatus; + +typedef struct { +	IconSVG **iconlist; +	int iconlist_num; +	eStatus status; +	 +	char *hidden_stop; +	char *hidden_start; +	char *publisher_stop; +	char *publisher_start; +	char *coordinate_stop; +	char *coordinate_start; +} OneCanvasIconInfo; + +/* + * Find the start of the next XML tag (search for '<') + */ +static inline char *xml_nextTag(char *c) +{ +	c++; +	if(c == NULL) +		return NULL; +	return strchr(c, '<'); +} + +/* + * Pull out the name/type of a tag. + */ +static inline char *xml_getTagName(char *c) +{ +	char *tag_end = NULL, *tag_space, *tag_close, *tag_feed, *tag_line; +	static char tagname[20]; +	int tag_len; +	 +	if(++c == NULL) +		return NULL; +	tag_space = strchr(c, ' '); +	tag_close = strchr(c, '>'); +	tag_feed = strchr(c, '\r'); +	tag_line = strchr(c, '\n'); +	if(tag_space) +		tag_end = tag_space; +	if(tag_close && tag_end > tag_close) +		tag_end = tag_close; +	if(tag_feed && tag_end > tag_feed) +		tag_end = tag_feed; +	if(tag_line && tag_end > tag_line) +		tag_end = tag_line; +	if(!tag_end) +		return NULL; +	tag_len = tag_end - c; +	tag_len = tag_len > 19 ? 19 : tag_len; +	strncpy(tagname, c, tag_len); +	tagname[tag_len] = '\0'; +	return tagname; +} + +/* + * Find the position in the string corresponding to a particular named attribute. + */ +static inline char *xml_getTagAttributePtr(char *c, char *attrname) +{ +	char *end, *name; +	int found; +	 +	if(++c == NULL) +		return NULL; +	end = strchr(c, '>'); +	while(c < end) +	{ +		int name_len; +		char *equal; + +		equal = c = strchr(c, '='); +		if(c == NULL) +			break; +		c++; +		name = equal; +		while(name[0] != ' ' && name[0] != '\t' && name[0] != '\n') +			name--; +		name++; /* don't include the space */ +		name_len = equal-name; +		if(name_len != strlen(attrname)) +			continue; +		if(strncasecmp(attrname, name, name_len) == 0) +		{ +			found = TRUE; +			break; +		} +	} +	if(!found) +		return NULL; +	return c-strlen(attrname)-1; +} + +/* + * Return the value of an XML tag's named attribute. + */ +static inline char *xml_getTagAttribute(char *c, char *attrname) +{ +	char *data_end; +	int data_len; +	char *attr; + +	c = xml_getTagAttributePtr(c, attrname); +	if(c == NULL) +		return NULL; +	c+=strlen(attrname); /* skip the name */ +	c+=2; /* skip the equals sign and the quote */ +	data_end = strchr(c, '"'); +	data_len = data_end - c; +	attr = (char *) malloc(data_len+1); +	strncpy(attr, c, data_len); +	attr[data_len] = '\0'; +	return attr; +} + +/* + * Find the value of an XML tag attribute and convert it to a number. + */  +static inline double xml_getTagAttributeFloat(char *c, char *attrname) +{ +	char *value = xml_getTagAttribute(c, attrname); +	double ret; + +	if(!value) +		return nan("nan"); +	sscanf(value, "%lf", &ret); +	free(value); +	return ret; +} + +/* + * Match the beginning an XML tag by "id" (preferred) or Inkscape's + * label (undesireable but acceptable). + */ +static inline char *xml_idMatchStart(char *stream_pos, char *layer_name) +{ +	char *id_acceptable = xml_getTagAttribute(stream_pos, "inkscape:label"); +	char *id_preferred = xml_getTagAttribute(stream_pos, "id"); + +	if(id_preferred && strncasecmp(id_preferred, layer_name, strlen(layer_name)) == 0) +	{ +		free(id_acceptable); +		return id_preferred; +	} +	if(id_acceptable && strncasecmp(id_acceptable, layer_name, strlen(layer_name)) == 0) +	{ +		free(id_preferred); +		return id_acceptable; +	} +	free(id_acceptable); +	free(id_preferred); +	return NULL; +} + +/* + * Match the entirety of an XML tag by "id" (preferred) or Inkscape's + * label (undesireable but acceptable). + */ +static inline int xml_idMatch(char *stream_pos, char *layer_name) +{ +	char *id_acceptable = xml_getTagAttribute(stream_pos, "inkscape:label"); +	char *id_preferred = xml_getTagAttribute(stream_pos, "id"); +	int ret = FALSE; + +	if((id_preferred && strcasecmp(id_preferred, layer_name) == 0) +	    || (id_acceptable && strcasecmp(id_acceptable, layer_name) == 0)) +		ret = TRUE; +	free(id_acceptable); +	free(id_preferred); +	return ret; +} + +/* + * Strip all the XML tags from a string and return only the data not + * contained within any tags. + */ +static inline char *xml_stripTags(char *data, int len) +{ +	char *ret = (char *) malloc(len+1); +	char *tag_left, *tag_right; + +	memcpy(ret, data, len+1); +	ret[len] = '\0'; +	while((tag_left = strchr(ret, '<')) != NULL) +	{ +		tag_right = strchr(ret, '>'); +		memmove(tag_left, tag_right+1, strlen(ret)-(tag_right-ret)); +	} +	return ret; +} + +/* + * Return the information for all of the icons within a "one-canvas" + * data stream. + */ +OneCanvasIconInfo onecanvas_geticons(char *stream) +{ +	eStatus status = STATUS_FINDSVG; +	unsigned int stream_size = 0; +	OneCanvasIconInfo info; +	char *publisher = NULL; +	char *stream_pos; +	int i; + +	memset(&info, 0, sizeof(info));  +	stream_pos = stream; +	while(stream_pos) +	{ +		char *name = xml_getTagName(stream_pos); + +		if(!name) +		{ +			stream_pos = xml_nextTag(stream_pos); +			continue; +		} +		switch(status) +		{ +			case STATUS_FINDSVG: +			{ +				if(strcasecmp(name, "svg") == 0) +				{ +					info.coordinate_start = xml_getTagAttributePtr(stream_pos, "x"); +					info.coordinate_stop = xml_getTagAttributePtr(stream_pos, "viewBox"); +					if(info.coordinate_start == NULL || info.coordinate_stop == NULL) +					{ +						status = STATUS_FAILED; +						break; +					} +					info.coordinate_stop = strchr(info.coordinate_stop, '"')+1; +					info.coordinate_stop = strchr(info.coordinate_stop, '"')+1; +					status = STATUS_FINDMETADATA; +				} +			} break; +			case STATUS_FINDMETADATA: +			{ +				if(strcasecmp(name, "metadata") == 0) +				{ +					status = STATUS_FINDPUBLISHER_START; +				} +				else if(strcasecmp(name, "/svg") == 0) +				{ +					status = STATUS_FAILED; +				} +			} break; +			case STATUS_FINDPUBLISHER_START: +			{ +				if(strcasecmp(name, "dc:publisher") == 0) +				{ +					status = STATUS_FINDPUBLISHER_STOP; +					info.publisher_start = stream_pos + strlen("<dc:publisher>"); +				} +				else if(strcasecmp(name, "/metadata") == 0) +				{ +					status = STATUS_FAILED; +				} +			} break; +			case STATUS_FINDPUBLISHER_STOP: +			{ +				if(strcasecmp(name, "/dc:publisher") == 0) +				{ +					info.publisher_stop = stream_pos; +					publisher = xml_stripTags(info.publisher_start, info.publisher_stop-info.publisher_start); +					if(strcasecmp(publisher, "one-canvas") == 0) +						status = STATUS_FINDHIDDEN; +					else +						status = STATUS_FAILED; +				} +				else if(strcasecmp(name, "/metadata") == 0) +				{ +					status = STATUS_FAILED; +				} +			} break; +			case STATUS_FINDHIDDEN: +			{ +				if(strcasecmp(name, "g") == 0) +				{ +					if(xml_idMatch(stream_pos, "hidden")) +					{ +						char *style_start; + +						info.hidden_start = stream_pos; +						info.hidden_stop = info.hidden_start; +						style_start = xml_getTagAttributePtr(stream_pos, "style"); +						if(style_start) +						{ +							info.hidden_start = style_start; +							info.hidden_stop = strchr(style_start, '"')+1; +							info.hidden_stop = strchr(info.hidden_stop, '"')+1; +						} +						else +						{ +							info.hidden_start += strlen("<g "); +							info.hidden_stop += strlen("<g "); +						} +						status = STATUS_FINDBOUNDS; +					} +				} +			} break; +			case STATUS_FINDBOUNDS: +			{ +				if(strcasecmp(name, "rect") == 0) +				{ +					char *layer_name = xml_idMatchStart(stream_pos, "iconlayer-"); + +					if(layer_name != NULL) +					{ +						IconSVG *icon = (IconSVG *) malloc(sizeof(IconSVG)); + +						icon->x = xml_getTagAttributeFloat(stream_pos, "x"); +						icon->y = xml_getTagAttributeFloat(stream_pos, "y"); +						icon->width = xml_getTagAttributeFloat(stream_pos, "width"); +						icon->height = xml_getTagAttributeFloat(stream_pos, "height"); +						sscanf(layer_name, "iconlayer-%dx%d", &(icon->icon_width), &(icon->icon_height)); +						free(layer_name); +						status = STATUS_FINDBOUNDS; +						info.iconlist = (IconSVG **) realloc(info.iconlist, (info.iconlist_num+1)*sizeof(IconSVG *)); +						info.iconlist[info.iconlist_num] = icon; +						info.iconlist_num++; +					} +				} +				else if(strcasecmp(name, "/g") == 0) +				{ +					status = STATUS_DONE; +				} +			} break; +			default: +				break; +		} +		if(status == STATUS_DONE || status == STATUS_FAILED) +			break; +		stream_pos = xml_nextTag(stream_pos); +	} +	free(publisher); +	info.status = status; +	return info; +} + +/* + * Obtain a single icon from the "one-canvas" stream corresponding + * to a particular icon size. + */ +char *onecanvas_geticon_bysize(char *icon_data, int requested_size) +{ +	OneCanvasIconInfo info = onecanvas_geticons(icon_data); +	char *ret = NULL; +	int i; +	 +	if(info.status == STATUS_DONE && info.iconlist_num > 0) +	{ +		int closest_diff = abs(info.iconlist[0]->icon_width - requested_size); +		int tocoord_length, topubl_length, tohidden_length; +		int icon_id = 0; +		IconSVG *icon; +		int ret_max; +		 +		for(i=0;i<info.iconlist_num;i++) +		{ +			int size_diff = abs(info.iconlist[i]->icon_width - requested_size); +			 +			if(size_diff < closest_diff) +			{ +				closest_diff = size_diff; +				icon_id = i; +			} +		} +		icon = info.iconlist[icon_id]; + 		/* Note: 200 characters is a very generous over estimate for the data we add in */ +		ret_max = strlen(icon_data)+1+200; +		ret = (char *) malloc(ret_max); +		tocoord_length = info.coordinate_start-icon_data; +		snprintf(ret, ret_max, "%.*s", tocoord_length, icon_data); +		/* Output the coordinates of the icon */ +		snprintf(&ret[strlen(ret)], ret_max-strlen(ret), "\nx=\"0px\"\ny=\"0px\"\n"); +		snprintf(&ret[strlen(ret)], ret_max-strlen(ret), "width=\"%d\"\n", icon->icon_width); +		snprintf(&ret[strlen(ret)], ret_max-strlen(ret), "height=\"%d\"\n", icon->icon_height); +		snprintf(&ret[strlen(ret)], ret_max-strlen(ret), "viewBox=\"%lf %lf %lf %lf\"\n", icon->x, icon->y, icon->width, icon->height); +		topubl_length = info.publisher_start-info.coordinate_stop; +		snprintf(&ret[strlen(ret)], ret_max-strlen(ret), "%.*s", topubl_length, info.coordinate_stop); +		/* Hide the "hidden" layer */ +		tohidden_length = info.hidden_start-info.publisher_stop; +		snprintf(&ret[strlen(ret)], ret_max-strlen(ret), "%.*s", tohidden_length, info.publisher_stop); +		snprintf(&ret[strlen(ret)], ret_max-strlen(ret), "\ndisplay=\"none\"\n"); +		/* Output the rest of the document */ +		snprintf(&ret[strlen(ret)], ret_max-strlen(ret), "%s", info.hidden_stop); +	} +	for(i=0;i<info.iconlist_num;i++) +		free(info.iconlist[i]); +	free(info.iconlist); +	return ret; +}
\ No newline at end of file diff --git a/src/onecanvas.h b/src/onecanvas.h new file mode 100644 index 0000000..e201417 --- /dev/null +++ b/src/onecanvas.h @@ -0,0 +1,6 @@ +#ifndef __ONECANVAS_H +#define __ONECANVAS_H + +char *onecanvas_geticon_bysize(char *icon_data, int requested_size); + +#endif /* __ONECANVAS_H */ diff --git a/src/tempfiles.c b/src/tempfiles.c new file mode 100644 index 0000000..edf72a1 --- /dev/null +++ b/src/tempfiles.c @@ -0,0 +1,317 @@ +/* + * + *  Copyright (c) 2009 Erich Hoover + * + *  libr temp files - Handle temporary files and handles that require cleanup + *                    when libr closes. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 2.1 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * 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 + * + * To provide feedback, report bugs, or otherwise contact me: + * ehoover at mines dot edu + * + */ + +#include "tempfiles.h" + +/* For malloc/free and mkdtemp */ +#include <stdlib.h> + +/* For string handling */ +#include <string.h> +#include <stdio.h> + +/* For directory cleanup */ +#include <unistd.h> +#include <dirent.h> + +/* For directory creation */ +#include <sys/stat.h> +#include <sys/types.h> +#include <errno.h> + +#ifndef FALSE +#define FALSE 0 +#endif +#ifndef TRUE +#define TRUE 1 +#endif + +#ifndef DOXYGEN_SHOULD_SKIP_THIS + +/* Hold on to folder names for cleanup when libr is removed from memory */ +typedef struct CLEANUPFOLDER { +	char *folder; +	struct CLEANUPFOLDER *next; +} CleanupFolder; +CleanupFolder *folders_to_remove = NULL; + +/* Hold on to libr handles for cleanup when libr is removed from memory */ +typedef struct CLEANUPHANDLE { +	int internal; /* do not warn the user about cleaning this handle up */ +	libr_file *handle; +	struct CLEANUPHANDLE *next; +} CleanupHandle; +CleanupHandle *handles_to_remove = NULL; + +#endif /* DOXYGEN_SHOULD_SKIP_THIS */ + +/* + * Register a folder for cleanup when libr is removed from memory + */ +void register_folder_cleanup(char *temp_folder) +{ +	CleanupFolder *folder = malloc(sizeof(CleanupFolder)); +	 +	folder->folder = strdup(temp_folder); +	folder->next = NULL; +	if(folders_to_remove != NULL) +	{ +		CleanupFolder *f; +		 +		for(f = folders_to_remove; f->next != NULL; f = f->next) {} +		f->next = folder; +	} +	else +		folders_to_remove = folder; +} + +/* + * Register a libr handle for cleanup when libr is removed from memory + */ +void register_handle_cleanup(libr_file *handle) +{ +	CleanupHandle *h = malloc(sizeof(CleanupHandle)); +	 +	h->handle = handle; +	h->internal = FALSE; +	h->next = NULL; +	if(handles_to_remove != NULL) +	{ +		CleanupHandle *i; +		 +		for(i = handles_to_remove; i->next != NULL; i = i->next) {} +		i->next = h; +	} +	else +		handles_to_remove = h; +} + +/* + * Remove a libr handle from the cleanup list + */ +void unregister_handle_cleanup(libr_file *handle) +{ +	CleanupHandle *i, *last = NULL; +	int found = FALSE; + +	if(handles_to_remove == NULL) +	{ +		printf("Unregistering handle with no list of cleanup handles!\n"); +		return; +	} +	for(i = handles_to_remove; i != NULL; last = i, i = i->next) +	{ +		if(i->handle == handle) +		{ +			if(last == NULL) +				handles_to_remove = i->next; +			else +				last->next = i->next; +			free(i); +			found = TRUE; +			break; +		} +	} +	if(!found) +		printf("Could not find handle to remove from cleanup list!\n"); +} + +/* + * Flag a handle as internal (do not warn about unsafe cleanup) + */ +void register_internal_handle(libr_file *handle) +{ +	int found = FALSE; +	CleanupHandle *i; + +	if(handles_to_remove == NULL) +	{ +		printf("No cleanup list!\n"); +		return; +	} +	for(i = handles_to_remove; i != NULL; i = i->next) +	{ +		if(i->handle == handle) +		{ +			i->internal = TRUE; +			found = TRUE; +			break; +		} +	} +	if(!found) +		printf("Could not find handle in cleanup list!\n"); +} + +/* + * Cleanup a temporary folder used to hack the inability to load resources from a buffer + */ +void cleanup_folder(char *temp_folder) +{ +	char *filepath = (char *) malloc(PATH_MAX); +	DIR *dir = opendir(temp_folder); +	struct dirent *file; +	 +	while((file = readdir(dir)) != NULL) +	{ +		char *filename = file->d_name; +		 +		/* Do not delete "self" or "parent" directory entries */ +		if(!strcmp(filename, ".") || !strcmp(filename, "..")) +			continue; +		/* But delete anything else */ +		strcpy(filepath, temp_folder); +		strcat(filepath, "/"); +		strcat(filepath, filename); +		if(file->d_type == DT_DIR) +			cleanup_folder(filepath); +		else +		{ +			if(unlink(filepath)) +				printf("libr failed to cleanup '%s' in temporary folder: %m\n", filename); +		} +	} +	free(filepath); +	closedir(dir); +	if(rmdir(temp_folder) != 0) +		printf("libr failed to remove temporary folder: %m\n"); +} + +/* + * Perform cleanup when libr is removed from memory + */ +void do_cleanup(void) __attribute__((destructor)); +void do_cleanup(void) +{ +	CleanupFolder *f, *fnext; +	CleanupHandle *h, *hnext; +	 +	/* Cleanup folders */ +	for(f = folders_to_remove; f != NULL; f = fnext) +	{ +		folders_to_remove = NULL; +		fnext = f->next; +		cleanup_folder(f->folder); +		free(f->folder); +		free(f); +	} +	/* Cleanup handles */ +	for(h = handles_to_remove; h != NULL; h = hnext) +	{ +		handles_to_remove = NULL; +		hnext = h->next; +		/* Unless the handle was created internally then warn the developer to cleanup their act */ +		if(!h->internal) +			printf("Warning: Application did not cleanup resource handle: %p\n", h->handle); +		libr_close_internal(h->handle); +		free(h); +	} +} + +/* + * Build all the directories required by a resource + * (and construct the output string) + */ +int make_valid_path(char *out_path, size_t maxpath, char *start_folder, char *resource_name) +{ +	char *a, *c = resource_name; +	 +	strcpy(out_path, start_folder); +	while((a=strchr(c, '/')) != NULL) +	{ +		strcat(out_path, "/"); +		strncat(out_path, c, (size_t) (a-c)); +		if(mkdir(out_path, S_IRUSR|S_IWUSR|S_IXUSR) != 0) +		{ +			if(errno != EEXIST) +			{ +				printf("failed to make directory: %s %m\n", out_path); +				return false; +			} +		} +		c = a+1; +	} +	strcat(out_path, "/"); +	strcat(out_path, c); +	return true; +} + +/* + * Extract all the resources from the ELF file for use by the resource loader + */ +char *libr_extract_resources(libr_file *handle) +{ +	char *temp_mask = strdup(LIBR_TEMPFILE); +	char *temp_folder; +	int i = 0; +	 +	temp_folder = mkdtemp(temp_mask); +	if(temp_folder == NULL) +	{ +		/* failed to extract ELF resources, could not create a temporary path */ +		goto failed; +	} +	/* If this library cannot dynamically load resources then pull out all the resources to a temporary directory */ +	for(i=0;i<libr_resources(handle);i++) +	{ +		char *resource_name = libr_list(handle, i); +		char *file_path[PATH_MAX]; +		size_t resource_size; +		FILE *file_handle; +		char *resource; +		 +		resource = libr_malloc(handle, resource_name, &resource_size); +		if(!make_valid_path((char *)file_path, sizeof(file_path), temp_folder, resource_name)) +		{ +			/* failed to build the path required by a resource */ +			cleanup_folder(temp_folder); +			temp_folder = NULL; +			goto failed; +		} +		file_handle = fopen((const char *) file_path, "w"); +		if(file_handle == NULL) +		{ +			/* failed to extract ELF resources, could not write to temporary path */ +			cleanup_folder(temp_folder); +			temp_folder = NULL; +			goto failed; +		} +		/* if the resource is empty then fwrite will fail */ +		if( (resource_size != 0) && (fwrite(resource, resource_size, 1, file_handle) != 1) ) +		{ +			/* failed to extract ELF resources, temporary path out of space? */ +			cleanup_folder(temp_folder); +			temp_folder = NULL; +			goto failed; +		} +		fclose(file_handle); +		free(resource); +	} +failed: +	if(temp_folder != NULL) +		temp_folder = strdup(temp_folder); +	free(temp_mask); +	return temp_folder; +} diff --git a/src/tempfiles.h b/src/tempfiles.h new file mode 100644 index 0000000..5b9b0bc --- /dev/null +++ b/src/tempfiles.h @@ -0,0 +1,13 @@ +#ifndef __TEMPFILES_H +#define __TEMPFILES_H + +#include "libr.h" + +void cleanup_folder(char *temp_folder); +void register_handle_cleanup(libr_file *handle); +void unregister_handle_cleanup(libr_file *handle); +void register_internal_handle(libr_file *handle); +void register_folder_cleanup(char *temp_folder); +char *libr_extract_resources(libr_file *handle); + +#endif /* __TEMPFILES_H */ | 
