mirror of
				https://github.com/SoPat712/YTLitePlus.git
				synced 2025-10-30 20:34:03 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			291 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			291 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // Copyright (c) 2013, Facebook, Inc.
 | |
| // All rights reserved.
 | |
| // Redistribution and use in source and binary forms, with or without
 | |
| // modification, are permitted provided that the following conditions are met:
 | |
| //   * Redistributions of source code must retain the above copyright notice,
 | |
| //     this list of conditions and the following disclaimer.
 | |
| //   * Redistributions in binary form must reproduce the above copyright notice,
 | |
| //     this list of conditions and the following disclaimer in the documentation
 | |
| //     and/or other materials provided with the distribution.
 | |
| //   * Neither the name Facebook nor the names of its contributors may be used to
 | |
| //     endorse or promote products derived from this software without specific
 | |
| //     prior written permission.
 | |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 | |
| // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | |
| // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 | |
| // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
 | |
| // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 | |
| // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 | |
| // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
 | |
| // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 | |
| // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 | |
| // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | |
| 
 | |
| #include "flex_fishhook.h"
 | |
| 
 | |
| #include <dlfcn.h>
 | |
| #include <stdbool.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <sys/mman.h>
 | |
| #include <sys/types.h>
 | |
| #include <mach/mach.h>
 | |
| #include <mach/vm_map.h>
 | |
| #include <mach/vm_region.h>
 | |
| #include <mach-o/dyld.h>
 | |
| #include <mach-o/loader.h>
 | |
| #include <mach-o/nlist.h>
 | |
| 
 | |
| #ifdef __LP64__
 | |
| typedef struct mach_header_64 mach_header_t;
 | |
| typedef struct segment_command_64 segment_command_t;
 | |
| typedef struct section_64 section_t;
 | |
| typedef struct nlist_64 nlist_t;
 | |
| #define LC_SEGMENT_ARCH_DEPENDENT LC_SEGMENT_64
 | |
| #else
 | |
| typedef struct mach_header mach_header_t;
 | |
| typedef struct segment_command segment_command_t;
 | |
| typedef struct section section_t;
 | |
| typedef struct nlist nlist_t;
 | |
| #define LC_SEGMENT_ARCH_DEPENDENT LC_SEGMENT
 | |
| #endif
 | |
| 
 | |
| #ifndef SEG_DATA_CONST
 | |
| #define SEG_DATA_CONST  "__DATA_CONST"
 | |
| #endif
 | |
| 
 | |
| struct rebindings_entry {
 | |
|     struct rebinding *rebindings;
 | |
|     size_t rebindings_nel;
 | |
|     struct rebindings_entry *next;
 | |
| };
 | |
| 
 | |
| static struct rebindings_entry *_flex_rebindings_head;
 | |
| 
 | |
| /// @return 0 on success
 | |
| static int flex_prepend_rebindings(struct rebindings_entry **rebindings_head,
 | |
|                               struct rebinding rebindings[],
 | |
|                               size_t nel) {
 | |
|     struct rebindings_entry *new_entry = (struct rebindings_entry *) malloc(sizeof(struct rebindings_entry));
 | |
|     if (!new_entry) {
 | |
|         return -1;
 | |
|     }
 | |
|     
 | |
|     new_entry->rebindings = (struct rebinding *) malloc(sizeof(struct rebinding) * nel);
 | |
|     if (!new_entry->rebindings) {
 | |
|         free(new_entry);
 | |
|         return -1;
 | |
|     }
 | |
|     
 | |
|     memcpy(new_entry->rebindings, rebindings, sizeof(struct rebinding) * nel);
 | |
|     new_entry->rebindings_nel = nel;
 | |
|     new_entry->next = *rebindings_head;
 | |
|     *rebindings_head = new_entry;
 | |
|     
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static vm_prot_t flex_get_protection(void *sectionStart) {
 | |
|     mach_port_t task = mach_task_self();
 | |
|     vm_size_t size = 0;
 | |
|     vm_address_t address = (vm_address_t)sectionStart;
 | |
|     memory_object_name_t object;
 | |
| #if __LP64__
 | |
|     mach_msg_type_number_t count = VM_REGION_BASIC_INFO_COUNT_64;
 | |
|     vm_region_basic_info_data_64_t info;
 | |
|     kern_return_t info_ret = vm_region_64(
 | |
|         task, &address, &size, VM_REGION_BASIC_INFO_64,
 | |
|         (vm_region_info_64_t)&info, &count, &object
 | |
|     );
 | |
| #else
 | |
|     mach_msg_type_number_t count = VM_REGION_BASIC_INFO_COUNT;
 | |
|     vm_region_basic_info_data_t info;
 | |
|     kern_return_t info_ret = vm_region(
 | |
|         task, &address, &size, VM_REGION_BASIC_INFO, (vm_region_info_t)&info, &count, &object
 | |
|     );
 | |
| #endif
 | |
|     if (info_ret == KERN_SUCCESS) {
 | |
|         return info.protection;
 | |
|     } else {
 | |
|         return VM_PROT_READ;
 | |
|     }
 | |
| }
 | |
| static void flex_perform_rebinding_with_section(struct rebindings_entry *rebindings,
 | |
|                                                 section_t *section,
 | |
|                                                 intptr_t slide,
 | |
|                                                 nlist_t *symtab,
 | |
|                                                 char *strtab,
 | |
|                                                 uint32_t *indirect_symtab) {
 | |
|     const bool isDataConst = strcmp(section->segname, "__DATA_CONST") == 0;
 | |
|     uint32_t *indirect_symbol_indices = indirect_symtab + section->reserved1;
 | |
|     void **indirect_symbol_bindings = (void **)((uintptr_t)slide + section->addr);
 | |
|     vm_prot_t oldProtection = VM_PROT_READ;
 | |
|     
 | |
|     if (isDataConst) {
 | |
|         oldProtection = flex_get_protection(rebindings);
 | |
|         mprotect(indirect_symbol_bindings, section->size, PROT_READ | PROT_WRITE);
 | |
|     }
 | |
|     
 | |
|     for (uint i = 0; i < section->size / sizeof(void *); i++) {
 | |
|         uint32_t symtab_index = indirect_symbol_indices[i];
 | |
|         
 | |
|         if (symtab_index == INDIRECT_SYMBOL_ABS || symtab_index == INDIRECT_SYMBOL_LOCAL ||
 | |
|             symtab_index == (INDIRECT_SYMBOL_LOCAL   | INDIRECT_SYMBOL_ABS)) {
 | |
|             continue;
 | |
|         }
 | |
|         
 | |
|         uint32_t strtab_offset = symtab[symtab_index].n_un.n_strx;
 | |
|         char *symbol_name = strtab + strtab_offset;
 | |
|         bool symbol_name_longer_than_1 = symbol_name[0] && symbol_name[1];
 | |
|         struct rebindings_entry *cur = rebindings;
 | |
|         
 | |
|         while (cur) {
 | |
|             for (uint j = 0; j < cur->rebindings_nel; j++) {
 | |
|                 if (symbol_name_longer_than_1 &&
 | |
|                   strcmp(&symbol_name[1], cur->rebindings[j].name) == 0) {
 | |
|                     
 | |
|                     if (cur->rebindings[j].replaced != NULL &&
 | |
|                       indirect_symbol_bindings[i] != cur->rebindings[j].replacement) {
 | |
|                         
 | |
|                         *(cur->rebindings[j].replaced) = indirect_symbol_bindings[i];
 | |
|                     }
 | |
|                     
 | |
|                     indirect_symbol_bindings[i] = cur->rebindings[j].replacement;
 | |
|                     goto symbol_loop;
 | |
|                 }
 | |
|             }
 | |
|             
 | |
|             cur = cur->next;
 | |
|         }
 | |
|         
 | |
|     symbol_loop:;
 | |
|     }
 | |
|     
 | |
|     if (isDataConst) {
 | |
|         int protection = 0;
 | |
|         if (oldProtection & VM_PROT_READ) {
 | |
|             protection |= PROT_READ;
 | |
|         }
 | |
|         if (oldProtection & VM_PROT_WRITE) {
 | |
|             protection |= PROT_WRITE;
 | |
|         }
 | |
|         if (oldProtection & VM_PROT_EXECUTE) {
 | |
|             protection |= PROT_EXEC;
 | |
|         }
 | |
|         
 | |
|         mprotect(indirect_symbol_bindings, section->size, protection);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void flex_rebind_symbols_for_image(struct rebindings_entry *rebindings,
 | |
|                                           const struct mach_header *header,
 | |
|                                           intptr_t slide) {
 | |
|     Dl_info info;
 | |
|     if (dladdr(header, &info) == 0) {
 | |
|         return;
 | |
|     }
 | |
|     
 | |
|     segment_command_t *cur_seg_cmd;
 | |
|     segment_command_t *linkedit_segment = NULL;
 | |
|     struct symtab_command* symtab_cmd = NULL;
 | |
|     struct dysymtab_command* dysymtab_cmd = NULL;
 | |
|     
 | |
|     uintptr_t cur = (uintptr_t)header + sizeof(mach_header_t);
 | |
|     for (uint i = 0; i < header->ncmds; i++, cur += cur_seg_cmd->cmdsize) {
 | |
|         cur_seg_cmd = (segment_command_t *)cur;
 | |
|         
 | |
|         if (cur_seg_cmd->cmd == LC_SEGMENT_ARCH_DEPENDENT) {
 | |
|             if (strcmp(cur_seg_cmd->segname, SEG_LINKEDIT) == 0) {
 | |
|                 linkedit_segment = cur_seg_cmd;
 | |
|             }
 | |
|         } else if (cur_seg_cmd->cmd == LC_SYMTAB) {
 | |
|             symtab_cmd = (struct symtab_command*)cur_seg_cmd;
 | |
|         } else if (cur_seg_cmd->cmd == LC_DYSYMTAB) {
 | |
|             dysymtab_cmd = (struct dysymtab_command*)cur_seg_cmd;
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     if (!symtab_cmd || !dysymtab_cmd || !linkedit_segment ||
 | |
|         !dysymtab_cmd->nindirectsyms) {
 | |
|         return;
 | |
|     }
 | |
|     
 | |
|     // Find base symbol/string table addresses
 | |
|     uintptr_t linkedit_base = (uintptr_t)slide + linkedit_segment->vmaddr - linkedit_segment->fileoff;
 | |
|     nlist_t *symtab = (nlist_t *)(linkedit_base + symtab_cmd->symoff);
 | |
|     char *strtab = (char *)(linkedit_base + symtab_cmd->stroff);
 | |
|     
 | |
|     // Get indirect symbol table (array of uint32_t indices into symbol table)
 | |
|     uint32_t *indirect_symtab = (uint32_t *)(linkedit_base + dysymtab_cmd->indirectsymoff);
 | |
|     
 | |
|     cur = (uintptr_t)header + sizeof(mach_header_t);
 | |
|     for (uint i = 0; i < header->ncmds; i++, cur += cur_seg_cmd->cmdsize) {
 | |
|         cur_seg_cmd = (segment_command_t *)cur;
 | |
|         
 | |
|         if (cur_seg_cmd->cmd == LC_SEGMENT_ARCH_DEPENDENT) {
 | |
|             if (strcmp(cur_seg_cmd->segname, SEG_DATA) != 0 &&
 | |
|                 strcmp(cur_seg_cmd->segname, SEG_DATA_CONST) != 0) {
 | |
|                 continue;
 | |
|             }
 | |
|             
 | |
|             for (uint j = 0; j < cur_seg_cmd->nsects; j++) {
 | |
|                 section_t *sect = (section_t *)(cur + sizeof(segment_command_t)) + j;
 | |
|                 
 | |
|                 if ((sect->flags & SECTION_TYPE) == S_LAZY_SYMBOL_POINTERS) {
 | |
|                     flex_perform_rebinding_with_section(
 | |
|                         rebindings, sect, slide, symtab, strtab, indirect_symtab
 | |
|                     );
 | |
|                 }
 | |
|                 if ((sect->flags & SECTION_TYPE) == S_NON_LAZY_SYMBOL_POINTERS) {
 | |
|                     flex_perform_rebinding_with_section(
 | |
|                         rebindings, sect, slide, symtab, strtab, indirect_symtab
 | |
|                     );
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void _flex_rebind_symbols_for_image(const struct mach_header *header,
 | |
|                                            intptr_t slide) {
 | |
|     flex_rebind_symbols_for_image(_flex_rebindings_head, header, slide);
 | |
| }
 | |
| 
 | |
| int flex_rebind_symbols_image(void *header,
 | |
|                               intptr_t slide,
 | |
|                               struct rebinding rebindings[],
 | |
|                               size_t rebindings_nel) {
 | |
|     struct rebindings_entry *rebindings_head = NULL;
 | |
|     
 | |
|     int retval = flex_prepend_rebindings(&rebindings_head, rebindings, rebindings_nel);
 | |
|     flex_rebind_symbols_for_image(rebindings_head, (const struct mach_header *) header, slide);
 | |
|     
 | |
|     if (rebindings_head) {
 | |
|         free(rebindings_head->rebindings);
 | |
|     }
 | |
|     
 | |
|     free(rebindings_head);
 | |
|     return retval;
 | |
| }
 | |
| 
 | |
| /// @return 0 on success
 | |
| int flex_rebind_symbols(struct rebinding rebindings[], size_t rebindings_nel) {
 | |
|     int retval = flex_prepend_rebindings(&_flex_rebindings_head, rebindings, rebindings_nel);
 | |
|     if (retval < 0) {
 | |
|         return retval;
 | |
|     }
 | |
|     
 | |
|     // If this was the first call, register callback for image additions (which is also invoked for
 | |
|     // existing images, otherwise, just run on existing images
 | |
|     if (!_flex_rebindings_head->next) {
 | |
|         _dyld_register_func_for_add_image(_flex_rebind_symbols_for_image);
 | |
|     } else {
 | |
|         uint32_t c = _dyld_image_count();
 | |
|         for (uint32_t i = 0; i < c; i++) {
 | |
|             _flex_rebind_symbols_for_image(_dyld_get_image_header(i), _dyld_get_image_vmaddr_slide(i));
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     return retval;
 | |
| }
 | 
