]>
git.saurik.com Git - apple/ld64.git/blob - src/other/PruneTrie.cpp
766ae947696e5264b923e259b4e32d801dc6a43b
1 /* -*- mode: C++; c-basic-offset: 4; tab-width: 4 -*-
3 * Copyright (c) 2008 Apple Inc. All rights reserved.
5 * @APPLE_LICENSE_HEADER_START@
7 * This file contains Original Code and/or Modifications of Original Code
8 * as defined in and that are subject to the Apple Public Source License
9 * Version 2.0 (the 'License'). You may not use this file except in
10 * compliance with the License. Please obtain a copy of the License at
11 * http://www.opensource.apple.com/apsl/ and read it before using this
14 * The Original Code and all software distributed under the License are
15 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
16 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
17 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
19 * Please see the License for the specific language governing rights and
20 * limitations under the License.
22 * @APPLE_LICENSE_HEADER_END@
26 #include "MachOFileAbstraction.hpp"
27 #include "MachOTrie.hpp"
28 #include "prune_trie.h"
34 * prune_trie() is a C vended function that is used by strip(1) to prune out
35 * defined exported symbols from the export trie. It is passed a pointer to
36 * the start of bytes of the the trie and the size. The prune() funciton
37 * passed is called with each symbol name in the trie to determine if it is
38 * to be pruned (retuning 1) or not (returning 0). It writes the new trie
39 * back into the trie buffer and returns the new size in trie_new_size.
40 * If the pruning succeeds, NULL is returned. If there was an error processing
41 * the trie (e.f. it is malformed), then an error message string is returned.
42 * The error string can be freed.
47 uint32_t trie_start_size
,
48 int (*prune
)(const char *name
),
49 uint32_t* trie_new_size
)
51 // convert trie to vector of entries
52 std::vector
<mach_o::trie::Entry
> originalExports
;
54 parseTrie(trie_start
, trie_start
+trie_start_size
, originalExports
);
56 catch (const char* msg
) {
60 return strdup("unexpected exception processing trie");
63 // prune entries into new vector of entries
64 std::vector
<mach_o::trie::Entry
> newExports
;
65 newExports
.reserve(originalExports
.size());
66 for(std::vector
<mach_o::trie::Entry
>::iterator it
= originalExports
.begin(); it
!= originalExports
.end(); ++it
) {
67 if ( prune(it
->name
) == 0 )
68 newExports
.push_back(*it
);
71 // create new export trie
72 std::vector
<uint8_t> newExportTrieBytes
;
73 newExportTrieBytes
.reserve(trie_start_size
);
74 mach_o::trie::makeTrie(newExports
, newExportTrieBytes
);
75 // Need to align trie to 8 or 4 bytes. We don't know the arch, but if the incoming trie
76 // was not 8-byte aligned, then it can't be a 64-bit arch, so use 4-byte alignement.
77 if ( (trie_start_size
% 8) != 0 ) {
79 while ( (newExportTrieBytes
.size() % 4 ) != 0)
80 newExportTrieBytes
.push_back(0);
84 while ( (newExportTrieBytes
.size() % 8 ) != 0)
85 newExportTrieBytes
.push_back(0);
88 // copy into place, zero pad
89 *trie_new_size
= newExportTrieBytes
.size();
90 if ( *trie_new_size
> trie_start_size
) {
92 asprintf(&msg
, "new trie is larger (%d) than original (%d)", *trie_new_size
, trie_start_size
);
95 memcpy(trie_start
, &newExportTrieBytes
[0], *trie_new_size
);
96 bzero(trie_start
+*trie_new_size
, trie_start_size
- *trie_new_size
);