11dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project/* 25c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner * Copyright (C) 2008-2010 The Android Open Source Project 31dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * All rights reserved. 41dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * 51dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * Redistribution and use in source and binary forms, with or without 61dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * modification, are permitted provided that the following conditions 71dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * are met: 81dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * * Redistributions of source code must retain the above copyright 91dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * notice, this list of conditions and the following disclaimer. 101dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * * Redistributions in binary form must reproduce the above copyright 111dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * notice, this list of conditions and the following disclaimer in 121dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * the documentation and/or other materials provided with the 131dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * distribution. 141dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * 151dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 161dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 171dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 181dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE 191dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 201dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 211dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 221dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 231dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 241dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT 251dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 261dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * SUCH DAMAGE. 271dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project */ 281dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 291dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#ifndef _LINKER_DEBUG_H_ 301dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define _LINKER_DEBUG_H_ 311dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 321dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#include <stdio.h> 331dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 345c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#ifndef LINKER_DEBUG 355c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#error LINKER_DEBUG should be defined to either 1 or 0 in Android.mk 365c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#endif 375c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner 385c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner/* set LINKER_DEBUG_TO_LOG to 1 to send the logs to logcat, 395c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner * or 0 to use stdout instead. 405c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner */ 415c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#define LINKER_DEBUG_TO_LOG 1 425c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#define TRACE_DEBUG 1 435c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#define DO_TRACE_LOOKUP 1 445c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#define DO_TRACE_RELO 1 451dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define TIMING 0 461dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define STATS 0 471dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define COUNT_PAGES 0 481dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 491dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project/********************************************************************* 501dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * You shouldn't need to modify anything below unless you are adding 511dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * more debugging information. 521dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * 531dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * To enable/disable specific debug options, change the defines above 541dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project *********************************************************************/ 551dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 561dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 571dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project/*********************************************************************/ 581dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#undef TRUE 591dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#undef FALSE 601dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define TRUE 1 611dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define FALSE 0 621dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 631dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project/* Only use printf() during debugging. We have seen occasional memory 641dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project * corruption when the linker uses printf(). 651dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project */ 661dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#if LINKER_DEBUG 675c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#include "linker_format.h" 681dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Projectextern int debug_verbosity; 695c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#if LINKER_DEBUG_TO_LOG 705c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turnerextern int format_log(int, const char *, const char *, ...); 715c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#define _PRINTVF(v,f,x...) \ 725c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner do { \ 735c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner if (debug_verbosity > (v)) format_log(5-(v),"linker",x); \ 745c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner } while (0) 755c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#else /* !LINKER_DEBUG_TO_LOG */ 765c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turnerextern int format_fd(int, const char *, ...); 775c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#define _PRINTVF(v,f,x...) \ 785c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner do { \ 795c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner if (debug_verbosity > (v)) format_fd(1, x); \ 802e85579c34047c305caf15fb0ebe02bf3d001d0eDima Zavin } while (0) 815c734644eebf8d01be1e86cbe20a111a5c5a2738David 'Digit' Turner#endif /* !LINKER_DEBUG_TO_LOG */ 821dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#else /* !LINKER_DEBUG */ 831dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define _PRINTVF(v,f,x...) do {} while(0) 841dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#endif /* LINKER_DEBUG */ 851dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 861dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define PRINT(x...) _PRINTVF(-1, FALSE, x) 871dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define INFO(x...) _PRINTVF(0, TRUE, x) 881dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define TRACE(x...) _PRINTVF(1, TRUE, x) 891dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define WARN(fmt,args...) \ 901dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project _PRINTVF(-1, TRUE, "%s:%d| WARNING: " fmt, __FILE__, __LINE__, ## args) 912e85579c34047c305caf15fb0ebe02bf3d001d0eDima Zavin#define ERROR(fmt,args...) \ 922e85579c34047c305caf15fb0ebe02bf3d001d0eDima Zavin _PRINTVF(-1, TRUE, "%s:%d| ERROR: " fmt, __FILE__, __LINE__, ## args) 932e85579c34047c305caf15fb0ebe02bf3d001d0eDima Zavin 941dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 951dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#if TRACE_DEBUG 961dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define DEBUG(x...) _PRINTVF(2, TRUE, "DEBUG: " x) 971dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#else /* !TRACE_DEBUG */ 981dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define DEBUG(x...) do {} while (0) 991dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#endif /* TRACE_DEBUG */ 1001dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 1011dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#if LINKER_DEBUG 1021dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define TRACE_TYPE(t,x...) do { if (DO_TRACE_##t) { TRACE(x); } } while (0) 1031dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#else /* !LINKER_DEBUG */ 1041dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define TRACE_TYPE(t,x...) do {} while (0) 1051dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#endif /* LINKER_DEBUG */ 1061dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 1071dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#if STATS 1081dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define RELOC_ABSOLUTE 0 1091dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define RELOC_RELATIVE 1 1101dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define RELOC_COPY 2 1111dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define RELOC_SYMBOL 3 1121dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define NUM_RELOC_STATS 4 1131dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 1141dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Projectstruct _link_stats { 1151dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project int reloc[NUM_RELOC_STATS]; 1161dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project}; 1171dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Projectextern struct _link_stats linker_stats; 1181dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 1191dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define COUNT_RELOC(type) \ 1201dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project do { if (type >= 0 && type < NUM_RELOC_STATS) { \ 1211dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project linker_stats.reloc[type] += 1; \ 1221dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project } else { \ 1231dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project PRINT("Unknown reloc stat requested\n"); \ 1241dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project } \ 1251dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project } while(0) 1261dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#else /* !STATS */ 1271dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define COUNT_RELOC(type) do {} while(0) 1281dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#endif /* STATS */ 1291dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 1301dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#if TIMING 1311dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#undef WARN 1321dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define WARN(x...) do {} while (0) 1331dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#endif /* TIMING */ 1341dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 1351dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#if COUNT_PAGES 1361dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Projectextern unsigned bitmask[]; 1371dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define MARK(offset) do { \ 1381dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project bitmask[((offset) >> 12) >> 3] |= (1 << (((offset) >> 12) & 7)); \ 1391dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project } while(0) 1401dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#else 1411dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define MARK(x) do {} while (0) 1421dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#endif 1431dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 1441dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#define DEBUG_DUMP_PHDR(phdr, name, pid) do { \ 1451dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project DEBUG("%5d %s (phdr = 0x%08x)\n", (pid), (name), (unsigned)(phdr)); \ 1461dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project DEBUG("\t\tphdr->offset = 0x%08x\n", (unsigned)((phdr)->p_offset)); \ 1471dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project DEBUG("\t\tphdr->p_vaddr = 0x%08x\n", (unsigned)((phdr)->p_vaddr)); \ 1481dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project DEBUG("\t\tphdr->p_paddr = 0x%08x\n", (unsigned)((phdr)->p_paddr)); \ 1491dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project DEBUG("\t\tphdr->p_filesz = 0x%08x\n", (unsigned)((phdr)->p_filesz)); \ 1501dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project DEBUG("\t\tphdr->p_memsz = 0x%08x\n", (unsigned)((phdr)->p_memsz)); \ 1511dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project DEBUG("\t\tphdr->p_flags = 0x%08x\n", (unsigned)((phdr)->p_flags)); \ 1521dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project DEBUG("\t\tphdr->p_align = 0x%08x\n", (unsigned)((phdr)->p_align)); \ 1531dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project } while (0) 1541dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project 1551dc9e472e19acfe6dc7f41e429236e7eef7ceda1The Android Open Source Project#endif /* _LINKER_DEBUG_H_ */ 156