 2fa536d107
			
		
	
	
		2fa536d107
		
	
	
	
	
		
			
			This tool will be used for post-processing the linked vdso image, turning it into something that is easy to include into elfload.c. Reviewed-by: Philippe Mathieu-Daudé <philmd@linaro.org> Signed-off-by: Richard Henderson <richard.henderson@linaro.org>
		
			
				
	
	
		
			224 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			224 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Post-process a vdso elf image for inclusion into qemu.
 | |
|  *
 | |
|  * Copyright 2023 Linaro, Ltd.
 | |
|  *
 | |
|  * SPDX-License-Identifier: GPL-2.0-or-later
 | |
|  */
 | |
| 
 | |
| #include <stdlib.h>
 | |
| #include <stdbool.h>
 | |
| #include <stdint.h>
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <errno.h>
 | |
| #include <endian.h>
 | |
| #include <unistd.h>
 | |
| #include "elf.h"
 | |
| 
 | |
| 
 | |
| #define bswap_(p)  _Generic(*(p), \
 | |
|                             uint16_t: __builtin_bswap16,       \
 | |
|                             uint32_t: __builtin_bswap32,       \
 | |
|                             uint64_t: __builtin_bswap64,       \
 | |
|                             int16_t: __builtin_bswap16,        \
 | |
|                             int32_t: __builtin_bswap32,        \
 | |
|                             int64_t: __builtin_bswap64)
 | |
| #define bswaps(p) (*(p) = bswap_(p)(*(p)))
 | |
| 
 | |
| static void output_reloc(FILE *outf, void *buf, void *loc)
 | |
| {
 | |
|     fprintf(outf, "    0x%08tx,\n", loc - buf);
 | |
| }
 | |
| 
 | |
| static const char *sigreturn_sym;
 | |
| static const char *rt_sigreturn_sym;
 | |
| 
 | |
| static unsigned sigreturn_addr;
 | |
| static unsigned rt_sigreturn_addr;
 | |
| 
 | |
| #define N 32
 | |
| #define elfN(x)  elf32_##x
 | |
| #define ElfN(x)  Elf32_##x
 | |
| #include "gen-vdso-elfn.c.inc"
 | |
| #undef N
 | |
| #undef elfN
 | |
| #undef ElfN
 | |
| 
 | |
| #define N 64
 | |
| #define elfN(x)  elf64_##x
 | |
| #define ElfN(x)  Elf64_##x
 | |
| #include "gen-vdso-elfn.c.inc"
 | |
| #undef N
 | |
| #undef elfN
 | |
| #undef ElfN
 | |
| 
 | |
| 
 | |
| int main(int argc, char **argv)
 | |
| {
 | |
|     FILE *inf, *outf;
 | |
|     long total_len;
 | |
|     const char *prefix = "vdso";
 | |
|     const char *inf_name;
 | |
|     const char *outf_name = NULL;
 | |
|     unsigned char *buf;
 | |
|     bool need_bswap;
 | |
| 
 | |
|     while (1) {
 | |
|         int opt = getopt(argc, argv, "o:p:r:s:");
 | |
|         if (opt < 0) {
 | |
|             break;
 | |
|         }
 | |
|         switch (opt) {
 | |
|         case 'o':
 | |
|             outf_name = optarg;
 | |
|             break;
 | |
|         case 'p':
 | |
|             prefix = optarg;
 | |
|             break;
 | |
|         case 'r':
 | |
|             rt_sigreturn_sym = optarg;
 | |
|             break;
 | |
|         case 's':
 | |
|             sigreturn_sym = optarg;
 | |
|             break;
 | |
|         default:
 | |
|         usage:
 | |
|             fprintf(stderr, "usage: [-p prefix] [-r rt-sigreturn-name] "
 | |
|                     "[-s sigreturn-name] -o output-file input-file\n");
 | |
|             return EXIT_FAILURE;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (optind >= argc || outf_name == NULL) {
 | |
|         goto usage;
 | |
|     }
 | |
|     inf_name = argv[optind];
 | |
| 
 | |
|     /*
 | |
|      * Open the input and output files.
 | |
|      */
 | |
|     inf = fopen(inf_name, "rb");
 | |
|     if (inf == NULL) {
 | |
|         goto perror_inf;
 | |
|     }
 | |
|     outf = fopen(outf_name, "w");
 | |
|     if (outf == NULL) {
 | |
|         goto perror_outf;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * Read the input file into a buffer.
 | |
|      * We expect the vdso to be small, on the order of one page,
 | |
|      * therefore we do not expect a partial read.
 | |
|      */
 | |
|     fseek(inf, 0, SEEK_END);
 | |
|     total_len = ftell(inf);
 | |
|     fseek(inf, 0, SEEK_SET);
 | |
| 
 | |
|     buf = malloc(total_len);
 | |
|     if (buf == NULL) {
 | |
|         goto perror_inf;
 | |
|     }
 | |
| 
 | |
|     errno = 0;
 | |
|     if (fread(buf, 1, total_len, inf) != total_len) {
 | |
|         if (errno) {
 | |
|             goto perror_inf;
 | |
|         }
 | |
|         fprintf(stderr, "%s: incomplete read\n", inf_name);
 | |
|         return EXIT_FAILURE;
 | |
|     }
 | |
|     fclose(inf);
 | |
| 
 | |
|     /*
 | |
|      * Write out the vdso image now, before we make local changes.
 | |
|      */
 | |
| 
 | |
|     fprintf(outf,
 | |
|             "/* Automatically generated from linux-user/gen-vdso.c. */\n"
 | |
|             "\n"
 | |
|             "static const uint8_t %s_image[] = {",
 | |
|             prefix);
 | |
|     for (long i = 0; i < total_len; ++i) {
 | |
|         if (i % 12 == 0) {
 | |
|             fputs("\n   ", outf);
 | |
|         }
 | |
|         fprintf(outf, " 0x%02x,", buf[i]);
 | |
|     }
 | |
|     fprintf(outf, "\n};\n\n");
 | |
| 
 | |
|     /*
 | |
|      * Identify which elf flavor we're processing.
 | |
|      * The first 16 bytes of the file are e_ident.
 | |
|      */
 | |
| 
 | |
|     if (buf[EI_MAG0] != ELFMAG0 || buf[EI_MAG1] != ELFMAG1 ||
 | |
|         buf[EI_MAG2] != ELFMAG2 || buf[EI_MAG3] != ELFMAG3) {
 | |
|         fprintf(stderr, "%s: not an elf file\n", inf_name);
 | |
|         return EXIT_FAILURE;
 | |
|     }
 | |
|     switch (buf[EI_DATA]) {
 | |
|     case ELFDATA2LSB:
 | |
|         need_bswap = BYTE_ORDER != LITTLE_ENDIAN;
 | |
|         break;
 | |
|     case ELFDATA2MSB:
 | |
|         need_bswap = BYTE_ORDER != BIG_ENDIAN;
 | |
|         break;
 | |
|     default:
 | |
|         fprintf(stderr, "%s: invalid elf EI_DATA (%u)\n",
 | |
|                 inf_name, buf[EI_DATA]);
 | |
|         return EXIT_FAILURE;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * We need to relocate the VDSO image.  The one built into the kernel
 | |
|      * is built for a fixed address.  The one we built for QEMU is not,
 | |
|      * since that requires close control of the guest address space.
 | |
|      *
 | |
|      * Output relocation addresses as we go.
 | |
|      */
 | |
| 
 | |
|     fprintf(outf, "static const unsigned %s_relocs[] = {\n", prefix);
 | |
| 
 | |
|     switch (buf[EI_CLASS]) {
 | |
|     case ELFCLASS32:
 | |
|         elf32_process(outf, buf, need_bswap);
 | |
|         break;
 | |
|     case ELFCLASS64:
 | |
|         elf64_process(outf, buf, need_bswap);
 | |
|         break;
 | |
|     default:
 | |
|         fprintf(stderr, "%s: invalid elf EI_CLASS (%u)\n",
 | |
|                 inf_name, buf[EI_CLASS]);
 | |
|         return EXIT_FAILURE;
 | |
|     }
 | |
| 
 | |
|     fprintf(outf, "};\n\n");   /* end vdso_relocs. */
 | |
| 
 | |
|     fprintf(outf, "static const VdsoImageInfo %s_image_info = {\n", prefix);
 | |
|     fprintf(outf, "    .image = %s_image,\n", prefix);
 | |
|     fprintf(outf, "    .relocs = %s_relocs,\n", prefix);
 | |
|     fprintf(outf, "    .image_size = sizeof(%s_image),\n", prefix);
 | |
|     fprintf(outf, "    .reloc_count = ARRAY_SIZE(%s_relocs),\n", prefix);
 | |
|     fprintf(outf, "    .sigreturn_ofs = 0x%x,\n", sigreturn_addr);
 | |
|     fprintf(outf, "    .rt_sigreturn_ofs = 0x%x,\n", rt_sigreturn_addr);
 | |
|     fprintf(outf, "};\n");
 | |
| 
 | |
|     /*
 | |
|      * Everything should have gone well.
 | |
|      */
 | |
|     if (fclose(outf)) {
 | |
|         goto perror_outf;
 | |
|     }
 | |
|     return EXIT_SUCCESS;
 | |
| 
 | |
|  perror_inf:
 | |
|     perror(inf_name);
 | |
|     return EXIT_FAILURE;
 | |
| 
 | |
|  perror_outf:
 | |
|     perror(outf_name);
 | |
|     return EXIT_FAILURE;
 | |
| }
 |