mirror of
https://github.com/yuzu-emu/breakpad.git
synced 2025-12-13 00:51:26 +00:00
To make it easier to add flags when adding new options in SymbolData. Example: I want to add a flag to disable inline record for https://chromium-review.googlesource.com/c/breakpad/breakpad/+/2915828. Change-Id: Ifc5da27c01efa0b0bc21cfcf769d4e6d604a63c6 Reviewed-on: https://chromium-review.googlesource.com/c/breakpad/breakpad/+/2984198 Reviewed-by: Joshua Peraza <jperaza@chromium.org>
468 lines
18 KiB
Plaintext
468 lines
18 KiB
Plaintext
// Copyright (c) 2006, Google 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 of Google Inc. 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
|
|
// OWNER 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.
|
|
|
|
// symupload.mm: Upload a symbol file to a HTTP server. The upload is sent as
|
|
// a multipart/form-data POST request with the following parameters:
|
|
// code_file: the basename of the module, e.g. "app"
|
|
// debug_file: the basename of the debugging file, e.g. "app"
|
|
// debug_identifier: the debug file's identifier, usually consisting of
|
|
// the guid and age embedded in the pdb, e.g.
|
|
// "11111111BBBB3333DDDD555555555555F"
|
|
// os: the operating system that the module was built for
|
|
// cpu: the CPU that the module was built for (x86 or ppc)
|
|
// symbol_file: the contents of the breakpad-format symbol file
|
|
|
|
#include <fcntl.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
|
|
#include <Foundation/Foundation.h>
|
|
|
|
#include "HTTPMultipartUpload.h"
|
|
#include "HTTPPutRequest.h"
|
|
#include "SymbolCollectorClient.h"
|
|
#include "common/mac/dump_syms.h"
|
|
|
|
using google_breakpad::DumpSymbols;
|
|
|
|
NSString* const kBreakpadSymbolType = @"BREAKPAD";
|
|
NSString* const kMachOSymbolType = @"MACHO";
|
|
NSString* const kDSYMSymbolType = @"DSYM";
|
|
|
|
typedef enum { kSymUploadProtocolV1, kSymUploadProtocolV2 } SymUploadProtocol;
|
|
|
|
typedef enum {
|
|
kResultSuccess = 0,
|
|
kResultFailure = 1,
|
|
kResultAlreadyExists = 2
|
|
} Result;
|
|
|
|
typedef struct {
|
|
NSString* symbolsPath;
|
|
NSString* uploadURLStr;
|
|
SymUploadProtocol symUploadProtocol;
|
|
NSString* apiKey;
|
|
BOOL force;
|
|
Result result;
|
|
NSString* type;
|
|
NSString* codeFile;
|
|
NSString* debugID;
|
|
} Options;
|
|
|
|
//=============================================================================
|
|
static NSArray* ModuleDataForSymbolFile(NSString* file) {
|
|
NSFileHandle* fh = [NSFileHandle fileHandleForReadingAtPath:file];
|
|
NSData* data = [fh readDataOfLength:1024];
|
|
NSString* str = [[NSString alloc] initWithData:data
|
|
encoding:NSUTF8StringEncoding];
|
|
NSScanner* scanner = [NSScanner scannerWithString:str];
|
|
NSString* line;
|
|
NSMutableArray* parts = nil;
|
|
const int MODULE_ID_INDEX = 3;
|
|
|
|
if ([scanner scanUpToString:@"\n" intoString:&line]) {
|
|
parts = [[NSMutableArray alloc] init];
|
|
NSScanner* moduleInfoScanner = [NSScanner scannerWithString:line];
|
|
NSString* moduleInfo;
|
|
// Get everything BEFORE the module name. None of these properties
|
|
// can have spaces.
|
|
for (int i = 0; i <= MODULE_ID_INDEX; i++) {
|
|
[moduleInfoScanner scanUpToString:@" " intoString:&moduleInfo];
|
|
[parts addObject:moduleInfo];
|
|
}
|
|
|
|
// Now get the module name. This can have a space so we scan to
|
|
// the end of the line.
|
|
[moduleInfoScanner scanUpToString:@"\n" intoString:&moduleInfo];
|
|
[parts addObject:moduleInfo];
|
|
}
|
|
|
|
[str release];
|
|
|
|
return parts;
|
|
}
|
|
|
|
//=============================================================================
|
|
static void StartSymUploadProtocolV1(Options* options,
|
|
NSString* OS,
|
|
NSString* CPU,
|
|
NSString* debugID,
|
|
NSString* debugFile) {
|
|
NSURL* url = [NSURL URLWithString:options->uploadURLStr];
|
|
HTTPMultipartUpload* ul = [[HTTPMultipartUpload alloc] initWithURL:url];
|
|
NSMutableDictionary* parameters = [NSMutableDictionary dictionary];
|
|
|
|
// Add parameters
|
|
[parameters setObject:debugID forKey:@"debug_identifier"];
|
|
[parameters setObject:OS forKey:@"os"];
|
|
[parameters setObject:CPU forKey:@"cpu"];
|
|
[parameters setObject:debugFile forKey:@"debug_file"];
|
|
[parameters setObject:debugFile forKey:@"code_file"];
|
|
[ul setParameters:parameters];
|
|
|
|
NSArray* keys = [parameters allKeys];
|
|
int count = [keys count];
|
|
for (int i = 0; i < count; ++i) {
|
|
NSString* key = [keys objectAtIndex:i];
|
|
NSString* value = [parameters objectForKey:key];
|
|
fprintf(stdout, "'%s' = '%s'\n", [key UTF8String], [value UTF8String]);
|
|
}
|
|
|
|
// Add file
|
|
[ul addFileAtPath:options->symbolsPath name:@"symbol_file"];
|
|
|
|
// Send it
|
|
NSError* error = nil;
|
|
NSData* data = [ul send:&error];
|
|
NSString* result = [[NSString alloc] initWithData:data
|
|
encoding:NSUTF8StringEncoding];
|
|
int status = [[ul response] statusCode];
|
|
|
|
fprintf(stdout, "Send: %s\n",
|
|
error ? [[error description] UTF8String] : "No Error");
|
|
fprintf(stdout, "Response: %d\n", status);
|
|
fprintf(stdout, "Result: %lu bytes\n%s\n", (unsigned long)[data length],
|
|
[result UTF8String]);
|
|
|
|
[result release];
|
|
[ul release];
|
|
options->result = (!error && status == 200) ? kResultSuccess : kResultFailure;
|
|
}
|
|
|
|
//=============================================================================
|
|
static void StartSymUploadProtocolV2(Options* options,
|
|
NSString* debugID,
|
|
NSString* debugFile) {
|
|
options->result = kResultFailure;
|
|
|
|
// Only check status of BREAKPAD symbols, because the v2 protocol doesn't
|
|
// (yet) have a way to check status of other symbol types.
|
|
if (!options->force && [options->type isEqualToString:kBreakpadSymbolType]) {
|
|
SymbolStatus symbolStatus =
|
|
[SymbolCollectorClient checkSymbolStatusOnServer:options->uploadURLStr
|
|
withAPIKey:options->apiKey
|
|
withDebugFile:debugFile
|
|
withDebugID:debugID];
|
|
if (symbolStatus == SymbolStatusFound) {
|
|
fprintf(stdout, "Symbol file already exists, upload aborted."
|
|
" Use \"-f\" to overwrite.\n");
|
|
options->result = kResultAlreadyExists;
|
|
return;
|
|
} else if (symbolStatus == SymbolStatusUnknown) {
|
|
fprintf(stdout, "Failed to get check for existing symbol.\n");
|
|
return;
|
|
}
|
|
}
|
|
|
|
UploadURLResponse* URLResponse =
|
|
[SymbolCollectorClient createUploadURLOnServer:options->uploadURLStr
|
|
withAPIKey:options->apiKey];
|
|
if (URLResponse == nil) {
|
|
return;
|
|
}
|
|
|
|
NSURL* uploadURL = [NSURL URLWithString:[URLResponse uploadURL]];
|
|
HTTPPutRequest* putRequest = [[HTTPPutRequest alloc] initWithURL:uploadURL];
|
|
[putRequest setFile:options->symbolsPath];
|
|
|
|
NSError* error = nil;
|
|
NSData* data = [putRequest send:&error];
|
|
NSString* result = [[NSString alloc] initWithData:data
|
|
encoding:NSUTF8StringEncoding];
|
|
int responseCode = [[putRequest response] statusCode];
|
|
[putRequest release];
|
|
|
|
if (error || responseCode != 200) {
|
|
fprintf(stdout, "Failed to upload symbol file.\n");
|
|
fprintf(stdout, "Response code: %d\n", responseCode);
|
|
fprintf(stdout, "Response:\n");
|
|
fprintf(stdout, "%s\n", [result UTF8String]);
|
|
return;
|
|
}
|
|
|
|
CompleteUploadResult completeUploadResult =
|
|
[SymbolCollectorClient completeUploadOnServer:options->uploadURLStr
|
|
withAPIKey:options->apiKey
|
|
withUploadKey:[URLResponse uploadKey]
|
|
withDebugFile:debugFile
|
|
withDebugID:debugID
|
|
withType:options->type];
|
|
[URLResponse release];
|
|
if (completeUploadResult == CompleteUploadResultError) {
|
|
fprintf(stdout, "Failed to complete upload.\n");
|
|
return;
|
|
} else if (completeUploadResult == CompleteUploadResultDuplicateData) {
|
|
fprintf(stdout, "Uploaded file checksum matched existing file checksum,"
|
|
" no change necessary.\n");
|
|
} else {
|
|
fprintf(stdout, "Successfully sent the symbol file.\n");
|
|
}
|
|
options->result = kResultSuccess;
|
|
}
|
|
|
|
//=============================================================================
|
|
static void Start(Options* options) {
|
|
// If non-BREAKPAD upload special-case.
|
|
if (![options->type isEqualToString:kBreakpadSymbolType]) {
|
|
StartSymUploadProtocolV2(options, options->debugID, options->codeFile);
|
|
return;
|
|
}
|
|
|
|
NSArray* moduleParts = ModuleDataForSymbolFile(options->symbolsPath);
|
|
// MODULE <os> <cpu> <uuid> <module-name>
|
|
// 0 1 2 3 4
|
|
NSString* OS = [moduleParts objectAtIndex:1];
|
|
NSString* CPU = [moduleParts objectAtIndex:2];
|
|
NSMutableString* debugID =
|
|
[NSMutableString stringWithString:[moduleParts objectAtIndex:3]];
|
|
[debugID replaceOccurrencesOfString:@"-"
|
|
withString:@""
|
|
options:0
|
|
range:NSMakeRange(0, [debugID length])];
|
|
NSString* debugFile = [moduleParts objectAtIndex:4];
|
|
|
|
if (options->symUploadProtocol == kSymUploadProtocolV1) {
|
|
StartSymUploadProtocolV1(options, OS, CPU, debugID, debugFile);
|
|
} else if (options->symUploadProtocol == kSymUploadProtocolV2) {
|
|
StartSymUploadProtocolV2(options, debugID, debugFile);
|
|
}
|
|
}
|
|
|
|
//=============================================================================
|
|
static void Usage(int argc, const char* argv[]) {
|
|
fprintf(stderr, "Submit symbol information.\n");
|
|
fprintf(stderr, "Usage: %s [options] <symbol-file> <upload-URL>\n", argv[0]);
|
|
fprintf(stderr, "<symbol-file> should be created by using the dump_syms "
|
|
"tool.\n");
|
|
fprintf(stderr, "<upload-URL> is the destination for the upload.\n");
|
|
fprintf(stderr, "Options:\n");
|
|
fprintf(stderr, "\t-p <protocol>: protocol to use for upload, accepts "
|
|
"[\"sym-upload-v1\", \"sym-upload-v2\"]. Default is "
|
|
"\"sym-upload-v1\".\n");
|
|
fprintf(stderr, "\t-k <api-key>: secret for authentication with upload "
|
|
"server. [Only in sym-upload-v2 protocol mode]\n");
|
|
fprintf(stderr, "\t-f: Overwrite symbol file on server if already present. "
|
|
"[Only in sym-upload-v2 protocol mode]\n");
|
|
fprintf(
|
|
stderr,
|
|
"-t:\t <symbol-type> Explicitly set symbol upload type ("
|
|
"default is 'breakpad').\n"
|
|
"\t One of ['breakpad', 'elf', 'pe', 'macho', 'debug_only', 'dwp', "
|
|
"'dsym', 'pdb'].\n"
|
|
"\t Note: When this flag is set to anything other than 'breakpad', then "
|
|
"the '-c' and '-i' flags must also be set.\n");
|
|
fprintf(stderr, "-c:\t <code-file> Explicitly set 'code_file' for symbol "
|
|
"upload (basename of executable).\n");
|
|
fprintf(stderr, "-i:\t <debug-id> Explicitly set 'debug_id' for symbol "
|
|
"upload (typically build ID of executable). The debug-id for "
|
|
"symbol-types 'dsym' and 'macho' will be determined "
|
|
"automatically. \n");
|
|
fprintf(stderr, "\t-h: Usage\n");
|
|
fprintf(stderr, "\t-?: Usage\n");
|
|
fprintf(stderr, "\n");
|
|
fprintf(stderr, "Exit codes:\n");
|
|
fprintf(stderr, "\t%d: Success\n", kResultSuccess);
|
|
fprintf(stderr, "\t%d: Failure\n", kResultFailure);
|
|
fprintf(stderr,
|
|
"\t%d: Symbol file already exists on server (and -f was not "
|
|
"specified).\n",
|
|
kResultAlreadyExists);
|
|
fprintf(stderr,
|
|
"\t [This exit code will only be returned by the sym-upload-v2 "
|
|
"protocol.\n");
|
|
fprintf(stderr,
|
|
"\t The sym-upload-v1 protocol can return either Success or "
|
|
"Failure\n");
|
|
fprintf(stderr, "\t in this case, and the action taken by the server is "
|
|
"unspecified.]\n");
|
|
fprintf(stderr, "\n");
|
|
fprintf(stderr, "Examples:\n");
|
|
fprintf(stderr, " With 'sym-upload-v1':\n");
|
|
fprintf(stderr, " %s path/to/symbol_file http://myuploadserver\n",
|
|
argv[0]);
|
|
fprintf(stderr, " With 'sym-upload-v2':\n");
|
|
fprintf(stderr, " [Defaulting to symbol type 'BREAKPAD']\n");
|
|
fprintf(stderr,
|
|
" %s -p sym-upload-v2 -k mysecret123! "
|
|
"path/to/symbol_file http://myuploadserver\n",
|
|
argv[0]);
|
|
fprintf(stderr, " [Explicitly set symbol type to 'macho']\n");
|
|
fprintf(stderr,
|
|
" %s -p sym-upload-v2 -k mysecret123! -t macho "
|
|
"-c app -i 11111111BBBB3333DDDD555555555555F "
|
|
"path/to/symbol_file http://myuploadserver\n",
|
|
argv[0]);
|
|
}
|
|
|
|
//=============================================================================
|
|
static void SetupOptions(int argc, const char* argv[], Options* options) {
|
|
// Set default options values.
|
|
options->symUploadProtocol = kSymUploadProtocolV1;
|
|
options->apiKey = nil;
|
|
options->type = kBreakpadSymbolType;
|
|
options->codeFile = nil;
|
|
options->debugID = nil;
|
|
options->force = NO;
|
|
|
|
extern int optind;
|
|
char ch;
|
|
|
|
while ((ch = getopt(argc, (char* const*)argv, "p:k:t:c:i:hf?")) != -1) {
|
|
switch (ch) {
|
|
case 'p':
|
|
if (strcmp(optarg, "sym-upload-v2") == 0) {
|
|
options->symUploadProtocol = kSymUploadProtocolV2;
|
|
break;
|
|
} else if (strcmp(optarg, "sym-upload-v1") == 0) {
|
|
// This is already the default but leave in case that changes.
|
|
options->symUploadProtocol = kSymUploadProtocolV1;
|
|
break;
|
|
}
|
|
Usage(argc, argv);
|
|
exit(0);
|
|
break;
|
|
case 'k':
|
|
options->apiKey = [NSString stringWithCString:optarg
|
|
encoding:NSASCIIStringEncoding];
|
|
break;
|
|
case 't': {
|
|
// This is really an enum, so treat as upper-case for consistency with
|
|
// enum naming convention on server-side.
|
|
options->type = [[NSString stringWithCString:optarg
|
|
encoding:NSASCIIStringEncoding]
|
|
uppercaseString];
|
|
break;
|
|
}
|
|
case 'c':
|
|
options->codeFile = [NSString stringWithCString:optarg
|
|
encoding:NSASCIIStringEncoding];
|
|
;
|
|
break;
|
|
case 'i':
|
|
options->debugID = [NSString stringWithCString:optarg
|
|
encoding:NSASCIIStringEncoding];
|
|
;
|
|
break;
|
|
case 'f':
|
|
options->force = YES;
|
|
break;
|
|
default:
|
|
Usage(argc, argv);
|
|
exit(0);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ((argc - optind) != 2) {
|
|
fprintf(stderr, "%s: Missing symbols file and/or upload-URL\n", argv[0]);
|
|
Usage(argc, argv);
|
|
exit(1);
|
|
}
|
|
|
|
int fd = open(argv[optind], O_RDONLY);
|
|
if (fd < 0) {
|
|
fprintf(stderr, "%s: %s: %s\n", argv[0], argv[optind], strerror(errno));
|
|
exit(1);
|
|
}
|
|
|
|
struct stat statbuf;
|
|
if (fstat(fd, &statbuf) < 0) {
|
|
fprintf(stderr, "%s: %s: %s\n", argv[0], argv[optind], strerror(errno));
|
|
close(fd);
|
|
exit(1);
|
|
}
|
|
close(fd);
|
|
|
|
if (!S_ISREG(statbuf.st_mode)) {
|
|
fprintf(stderr, "%s: %s: not a regular file\n", argv[0], argv[optind]);
|
|
exit(1);
|
|
}
|
|
|
|
bool isBreakpadUpload = [options->type isEqualToString:kBreakpadSymbolType];
|
|
bool hasCodeFile = options->codeFile != nil;
|
|
bool hasDebugID = options->debugID != nil;
|
|
if (isBreakpadUpload && (hasCodeFile || hasDebugID)) {
|
|
fprintf(stderr, "\n");
|
|
fprintf(stderr,
|
|
"%s: -c and -i should only be specified for non-breakpad "
|
|
"symbol upload types.\n",
|
|
argv[0]);
|
|
fprintf(stderr, "\n");
|
|
Usage(argc, argv);
|
|
exit(1);
|
|
}
|
|
|
|
if (!isBreakpadUpload && hasCodeFile && !hasDebugID &&
|
|
([options->type isEqualToString:kMachOSymbolType] ||
|
|
[options->type isEqualToString:kDSYMSymbolType])) {
|
|
DumpSymbols dump_symbols(SYMBOLS_AND_FILES | INLINES, false);
|
|
if (dump_symbols.Read(argv[optind])) {
|
|
std::string identifier = dump_symbols.Identifier();
|
|
if (identifier.empty()) {
|
|
fprintf(stderr, "\n");
|
|
fprintf(stderr,
|
|
"%s: Unable to determine debug-id. Please specify with '-i'.\n",
|
|
argv[0]);
|
|
fprintf(stderr, "\n");
|
|
Usage(argc, argv);
|
|
exit(1);
|
|
}
|
|
options->debugID = [NSString stringWithUTF8String:identifier.c_str()];
|
|
hasDebugID = true;
|
|
}
|
|
}
|
|
|
|
if (!isBreakpadUpload && (!hasCodeFile || !hasDebugID)) {
|
|
fprintf(stderr, "\n");
|
|
fprintf(stderr,
|
|
"%s: -c and -i must be specified for non-breakpad "
|
|
"symbol upload types.\n",
|
|
argv[0]);
|
|
fprintf(stderr, "\n");
|
|
Usage(argc, argv);
|
|
exit(1);
|
|
}
|
|
|
|
options->symbolsPath = [NSString stringWithUTF8String:argv[optind]];
|
|
options->uploadURLStr = [NSString stringWithUTF8String:argv[optind + 1]];
|
|
}
|
|
|
|
//=============================================================================
|
|
int main(int argc, const char* argv[]) {
|
|
NSAutoreleasePool* pool = [[NSAutoreleasePool alloc] init];
|
|
Options options;
|
|
|
|
bzero(&options, sizeof(Options));
|
|
SetupOptions(argc, argv, &options);
|
|
Start(&options);
|
|
|
|
[pool release];
|
|
return options.result;
|
|
}
|