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