UNPKG

asmimproved-dbgmits

Version:

Provides the ability to control GDB and LLDB programmatically via GDB/MI.

319 lines 20.1 kB
// Copyright (c) 2015 Vadim Macagon // MIT License, see LICENSE file for full terms. "use strict"; require('source-map-support').install(); var dbgmits = require('../lib/index'); var bunyan = require('bunyan'); var fs = require('fs'); var path = require('path'); var PrettyStream = require('bunyan-prettystream'); function startDebugSession(logger) { var debuggerType = ('lldb' === process.env['DBGMITS_DEBUGGER']) ? dbgmits.DebuggerType.LLDB : dbgmits.DebuggerType.GDB; var debugSession = dbgmits.startDebugSession(debuggerType); if (logger) { debugSession.logger = logger; // log event data emitted by DebugSession var eventsToLog = [ dbgmits.EVENT_TARGET_RUNNING, dbgmits.EVENT_TARGET_STOPPED, dbgmits.EVENT_BREAKPOINT_HIT, dbgmits.EVENT_STEP_FINISHED, dbgmits.EVENT_FUNCTION_FINISHED, dbgmits.EVENT_SIGNAL_RECEIVED, dbgmits.EVENT_EXCEPTION_RECEIVED, dbgmits.EVENT_THREAD_GROUP_ADDED, dbgmits.EVENT_THREAD_GROUP_REMOVED, dbgmits.EVENT_THREAD_GROUP_STARTED, dbgmits.EVENT_THREAD_GROUP_EXITED, dbgmits.EVENT_THREAD_CREATED, dbgmits.EVENT_THREAD_EXITED, dbgmits.EVENT_THREAD_SELECTED, dbgmits.EVENT_LIB_LOADED, dbgmits.EVENT_LIB_UNLOADED, ]; eventsToLog.forEach(function (eventName) { debugSession.on(eventName, function (data) { if (debugSession.logger) { debugSession.logger.debug({ event: eventName, data: data }); } }); }); // monkey-patch DebugSession methods that return non-void promises and log the values // the promises are resolved with var functionsToLog = [ 'addBreakpoint', 'ignoreBreakpoint', 'getStackFrame', 'getStackDepth', 'getStackFrames', 'getStackFrameArgs', 'getStackFrameVariables', 'addWatch', 'updateWatch', 'getWatchChildren', 'setWatchValueFormat', 'getWatchValue', 'setWatchValue', 'getWatchAttributes', 'getWatchExpression', 'evaluateExpression', 'readMemory', 'getRegisterNames', 'getRegisterValues', 'disassembleAddressRange', 'disassembleAddressRangeByLine', 'disassembleFile', 'disassembleFileByLine', 'getThread', 'getThreads' ]; functionsToLog.forEach(function (funcName) { var func = debugSession[funcName]; debugSession[funcName] = function () { return func.apply(this, arguments) .then(function (result) { if (debugSession.logger) { debugSession.logger.debug({ func: funcName, result: result }); } return result; }); }; }); } return debugSession; } exports.startDebugSession = startDebugSession; /** * This function performs the following tasks asynchronously (but sequentially): * 1. Adds a breakpoint on the given function. * 2. Runs the target until the breakpoint is hit. * 3. Invokes a callback. * * @param onBreakHit Callback to invoke after the specified function is reached. */ function runToFunc(debugSession, funcName, onBreakHit) { var onBreakpointHit = new Promise(function (resolve, reject) { debugSession.once(dbgmits.EVENT_BREAKPOINT_HIT, function (breakNotify) { onBreakHit() .then(resolve) .catch(reject); }); }); // add breakpoint to get to the starting point return debugSession.addBreakpoint(funcName) .then(function () { return Promise.all([ onBreakpointHit, debugSession.startInferior() ]); }); } exports.runToFunc = runToFunc; /** * This function performs the following tasks asynchronously (but sequentially): * 1. Adds a breakpoint on the given function. * 2. Runs the target until the breakpoint is hit. * 3. Steps out of the function within which the breakpoint was hit. * 4. Invokes a callback. * * @param afterStepOut Callback to invoke after the debugger finishes stepping out of the specified * function. */ function runToFuncAndStepOut(debugSession, funcName, afterStepOut) { var onStepOutRunTest = function () { return new Promise(function (resolve, reject) { if (debugSession.canEmitFunctionFinishedNotification()) { debugSession.once(dbgmits.EVENT_FUNCTION_FINISHED, function (stepNotify) { afterStepOut() .then(resolve) .catch(reject); }); } else { // FIXME: LLDB-MI currently doesn't emit a distinct notification for step-out so we have // to listen to the generic step-finished one. debugSession.once(dbgmits.EVENT_STEP_FINISHED, function (stepNotify) { afterStepOut() .then(resolve) .catch(reject); }); } }); }; var onBreakpointStepOut = new Promise(function (resolve, reject) { debugSession.once(dbgmits.EVENT_BREAKPOINT_HIT, function (breakNotify) { Promise.all([ onStepOutRunTest(), debugSession.stepOut() ]) .then(function () { resolve(); }) .catch(reject); }); }); // add breakpoint to get to the starting point return debugSession.addBreakpoint(funcName) .then(function () { return Promise.all([ onBreakpointStepOut, debugSession.startInferior() ]); }); } exports.runToFuncAndStepOut = runToFuncAndStepOut; ; /** * Wraps mocha's beforeEach() function so that a logger can be passed through to the callback * function `fn`. * * The logger instance passed to the `fn` function is created by a custom mocha reporter, * but only for tests that are in suites wrapped with [[logSuite]], or for tests wrapped with * [[logTest]]. * * @param fn A function to execute before each test in the current suite. */ function beforeEachTestWithLogger(fn) { // unfortunately mocha doesn't return the hook instance from beforeEach(), so it's not possible // to add additional fields to the hook itself, instead we have to add the additional fields // to the callback passed to beforeEach() var cb = function () { // the custom reporter should've already called cb.setLogger() by this stage return fn(cb.logger); }; cb.setLogger = function (logger) { cb.logger = logger; }; beforeEach(cb); } exports.beforeEachTestWithLogger = beforeEachTestWithLogger; /** * Creates a new logger for a test. * * The logger will create a log file in the `logs/tests` directory. * * @param testIndex Test identifier, should be unique for each test. * @param title Test title. * @return A new logger instance. */ function createLogger(testIndex, title) { try { fs.mkdirSync('logs/tests'); } catch (err) { if (err.code !== 'EEXIST') { throw err; } } // pad the test number with zeroes (e.g. 1 -> 001, 10 -> 010, 100 -> 100) var pad = '000'; var testFilename = (pad + testIndex).slice(-pad.length) + '.log'; var logPath = path.join('logs/tests', testFilename); var fileStream = fs.createWriteStream(logPath, { flags: 'w' }); var prettyStream = new PrettyStream({ useColor: false }); prettyStream.pipe(fileStream); var logger = bunyan.createLogger({ name: 'Test ' + testIndex, streams: [{ level: 'debug', type: 'raw', stream: prettyStream }] }); logger.info('====== TEST #%d: %s ======', testIndex, title); return logger; } /** * Attaches a createLogger() function to a mocha.Test. * * For example: * ``` * describe("Thingie", () => { * beforeEachTestWithLogger((logger: bunyan.Logger) => { * ... * }); * * logTest(it("Does something", () => { * ... * })); * }); * ``` */ function logTest(test) { test.createLogger = createLogger; return test; } exports.logTest = logTest; /** * Attaches a createLogger() function to a mocha.Suite. * * For example: * ``` * logSuite(describe("Thingie", () => { * beforeEachTestWithLogger((logger: bunyan.Logger) => { * ... * }); * * it("Does something", () => { * ... * }); * })); * ``` */ function logSuite(suite) { suite.createLogger = createLogger; return suite; } exports.logSuite = logSuite; /** * Escapes a string so that it can be safely embedded in a regular expression. * * @param original The string to escape. * @return The escaped string. */ function escapeStringForRegExp(original) { return original.replace(/[-\/\\^$*+?.()|[\]{}]/g, '\\$&'); } /** * Helper class for obtaining the line numbers of relevant source lines in the C++ source files * of the target executables used in tests. */ var SourceLineResolver = (function () { function SourceLineResolver() { this._sourceLines = []; } SourceLineResolver.loadSourceFileSync = function (filename) { var sourceLineResolver = new SourceLineResolver(); sourceLineResolver.loadFileSync(filename); return sourceLineResolver; }; SourceLineResolver.prototype.loadFileSync = function (filename) { this._sourceLines = fs.readFileSync(filename, 'utf8').split('\n'); }; /** * Finds the line number of the first source line that matches the given regular expression. * * @param sourceLineRegExp The regular expressions to match source lines against, note that * the expression is matched line by line so it shouldn't be created * with the multi-line flag. * @return The line number of the first matching line in the source file, * or -1 if no matching lines were found. */ SourceLineResolver.prototype.getMatchingLineNumber = function (sourceLineRegExp) { for (var i = 0; i < this._sourceLines.length; ++i) { if (sourceLineRegExp.test(this._sourceLines[i])) { return i + 1; } } return -1; }; /** * Finds the line number at which a given single-line comment is located. * * @param singleLineComment The test of the single-line (prefixed by `//`) comment. * @return The line number of a single line comment in the source file, * or -1 if no such comment was found in the source file. */ SourceLineResolver.prototype.getCommentLineNumber = function (singleLineComment) { var escapedComment = escapeStringForRegExp(singleLineComment); var commentRegExp = new RegExp("^[\\s\\S]*//\\s*" + escapedComment); return this.getMatchingLineNumber(commentRegExp); }; return SourceLineResolver; }()); exports.SourceLineResolver = SourceLineResolver; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"test_utils.js","sourceRoot":"","sources":["../test/test_utils.ts"],"names":[],"mappings":"AAAA,mCAAmC;AACnC,gDAAgD;;AAEhD,OAAO,CAAC,oBAAoB,CAAC,CAAC,OAAO,EAAE,CAAC;AAExC,IAAY,OAAO,WAAM,cAAc,CAAC,CAAA;AACxC,IAAY,MAAM,WAAM,QAAQ,CAAC,CAAA;AACjC,IAAY,EAAE,WAAM,IAAI,CAAC,CAAA;AACzB,IAAY,IAAI,WAAM,MAAM,CAAC,CAAA;AAC7B,IAAO,YAAY,WAAW,qBAAqB,CAAC,CAAC;AAKrD,2BAAkC,MAAsB;IACtD,IAAM,YAAY,GAAG,CAAC,MAAM,KAAK,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC,GAAG,OAAO,CAAC,YAAY,CAAC,IAAI,GAAG,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC;IACzH,IAAI,YAAY,GAAiB,OAAO,CAAC,iBAAiB,CAAC,YAAY,CAAC,CAAC;IACzE,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC;QACX,YAAY,CAAC,MAAM,GAAG,MAAM,CAAC;QAE7B,yCAAyC;QACzC,IAAI,WAAW,GAAG;YAChB,OAAO,CAAC,oBAAoB;YAC5B,OAAO,CAAC,oBAAoB;YAC5B,OAAO,CAAC,oBAAoB;YAC5B,OAAO,CAAC,mBAAmB;YAC3B,OAAO,CAAC,uBAAuB;YAC/B,OAAO,CAAC,qBAAqB;YAC7B,OAAO,CAAC,wBAAwB;YAChC,OAAO,CAAC,wBAAwB;YAChC,OAAO,CAAC,0BAA0B;YAClC,OAAO,CAAC,0BAA0B;YAClC,OAAO,CAAC,yBAAyB;YACjC,OAAO,CAAC,oBAAoB;YAC5B,OAAO,CAAC,mBAAmB;YAC3B,OAAO,CAAC,qBAAqB;YAC7B,OAAO,CAAC,gBAAgB;YACxB,OAAO,CAAC,kBAAkB;SAC3B,CAAC;QACF,WAAW,CAAC,OAAO,CAAC,UAAC,SAAiB;YACpC,YAAY,CAAC,EAAE,CAAC,SAAS,EAAE,UAAC,IAAS;gBACnC,EAAE,CAAC,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC;oBACxB,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,KAAK,EAAE,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;gBAC9D,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,qFAAqF;QACrF,iCAAiC;QACjC,IAAI,cAAc,GAAa;YAC7B,eAAe;YACf,kBAAkB;YAClB,eAAe;YACf,eAAe;YACf,gBAAgB;YAChB,mBAAmB;YACnB,wBAAwB;YACxB,UAAU;YACV,aAAa;YACb,kBAAkB;YAClB,qBAAqB;YACrB,eAAe;YACf,eAAe;YACf,oBAAoB;YACpB,oBAAoB;YACpB,oBAAoB;YACpB,YAAY;YACZ,kBAAkB;YAClB,mBAAmB;YACnB,yBAAyB;YACzB,+BAA+B;YAC/B,iBAAiB;YACjB,uBAAuB;YACvB,WAAW;YACX,YAAY;SACb,CAAC;QACF,cAAc,CAAC,OAAO,CAAC,UAAC,QAAgB;YACtC,IAAI,IAAI,GAAoB,YAAa,CAAC,QAAQ,CAAC,CAAC;YAC7C,YAAa,CAAC,QAAQ,CAAC,GAAG;gBAC/B,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,SAAS,CAAC;qBACjC,IAAI,CAAC,UAAC,MAAW;oBAChB,EAAE,CAAC,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC;wBACxB,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;oBAChE,CAAC;oBACD,MAAM,CAAC,MAAM,CAAC;gBAChB,CAAC,CAAC,CAAC;YACL,CAAC,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IACD,MAAM,CAAC,YAAY,CAAC;AACtB,CAAC;AA5Ee,yBAAiB,oBA4EhC,CAAA;AAED;;;;;;;GAOG;AACH,mBACE,YAA0B,EAAE,QAAgB,EAAE,UAA8B;IAE5E,IAAI,eAAe,GAAG,IAAI,OAAO,CAAO,UAAC,OAAO,EAAE,MAAM;QACtD,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,oBAAoB,EAC5C,UAAC,WAAwC;YACvC,UAAU,EAAE;iBACX,IAAI,CAAC,OAAO,CAAC;iBACb,KAAK,CAAC,MAAM,CAAC,CAAC;QACjB,CAAC,CACF,CAAC;IACJ,CAAC,CAAC,CAAC;IACH,8CAA8C;IAC9C,MAAM,CAAC,YAAY,CAAC,aAAa,CAAC,QAAQ,CAAC;SACxC,IAAI,CAAC;QACN,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC;YACjB,eAAe;YACf,YAAY,CAAC,aAAa,EAAE;SAC7B,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC;AApBe,iBAAS,YAoBxB,CAAA;AAED;;;;;;;;;GASG;AACH,6BACE,YAA0B,EAAE,QAAgB,EAAE,YAAgC;IAC9E,IAAI,gBAAgB,GAAG;QACrB,MAAM,CAAC,IAAI,OAAO,CAAO,UAAC,OAAO,EAAE,MAAM;YACvC,EAAE,CAAC,CAAC,YAAY,CAAC,mCAAmC,EAAE,CAAC,CAAC,CAAC;gBACvD,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,uBAAuB,EAC/C,UAAC,UAAyC;oBACxC,YAAY,EAAE;yBACb,IAAI,CAAC,OAAO,CAAC;yBACb,KAAK,CAAC,MAAM,CAAC,CAAC;gBACjB,CAAC,CACF,CAAC;YACJ,CAAC;YAAC,IAAI,CAAC,CAAC;gBACN,wFAAwF;gBACxF,8CAA8C;gBAC9C,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,mBAAmB,EAC3C,UAAC,UAAsC;oBACrC,YAAY,EAAE;yBACb,IAAI,CAAC,OAAO,CAAC;yBACb,KAAK,CAAC,MAAM,CAAC,CAAC;gBACjB,CAAC,CACF,CAAC;YACJ,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC,CAAC;IACF,IAAI,mBAAmB,GAAG,IAAI,OAAO,CAAO,UAAC,OAAO,EAAE,MAAM;QAC1D,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,oBAAoB,EAC5C,UAAC,WAAwC;YACvC,OAAO,CAAC,GAAG,CAAC;gBACV,gBAAgB,EAAE;gBAClB,YAAY,CAAC,OAAO,EAAE;aACvB,CAAC;iBACD,IAAI,CAAC,cAAQ,OAAO,EAAE,CAAC,CAAC,CAAC,CAAC;iBAC1B,KAAK,CAAC,MAAM,CAAC,CAAC;QACjB,CAAC,CACF,CAAC;IACJ,CAAC,CAAC,CAAC;IACH,8CAA8C;IAC9C,MAAM,CAAC,YAAY,CAAC,aAAa,CAAC,QAAQ,CAAC;SAC1C,IAAI,CAAC;QACJ,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC;YACjB,mBAAmB;YACnB,YAAY,CAAC,aAAa,EAAE;SAC7B,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC;AA7Ce,2BAAmB,sBA6ClC,CAAA;AAUA,CAAC;AAEF;;;;;;;;;GASG;AACH,kCAAyC,EAAkC;IACzE,+FAA+F;IAC/F,4FAA4F;IAC5F,yCAAyC;IACzC,IAAI,EAAE,GAAkB;QACtB,4EAA4E;QAC5E,MAAM,CAAC,EAAE,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;IACvB,CAAC,CAAC;IACF,EAAE,CAAC,SAAS,GAAG,UAAC,MAAqB;QACnC,EAAE,CAAC,MAAM,GAAG,MAAM,CAAC;IACrB,CAAC,CAAC;IACF,UAAU,CAAC,EAAE,CAAC,CAAC;AACjB,CAAC;AAZe,gCAAwB,2BAYvC,CAAA;AAED;;;;;;;;GAQG;AACH,sBAAsB,SAAiB,EAAE,KAAa;IACpD,IAAI,CAAC;QACH,EAAE,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC;IAC7B,CAAE;IAAA,KAAK,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;QACb,EAAE,CAAC,CAAC,GAAG,CAAC,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC;YAC1B,MAAM,GAAG,CAAC;QACZ,CAAC;IACH,CAAC;IACD,yEAAyE;IACzE,IAAI,GAAG,GAAG,KAAK,CAAC;IAChB,IAAI,YAAY,GAAG,CAAC,GAAG,GAAG,SAAS,CAAC,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,GAAG,MAAM,CAAC;IACjE,IAAI,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,YAAY,CAAC,CAAC;IACpD,IAAI,UAAU,GAAG,EAAE,CAAC,iBAAiB,CAAC,OAAO,EAAE,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,CAAC;IAC/D,IAAI,YAAY,GAAG,IAAI,YAAY,CAAC,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC,CAAC;IACzD,YAAY,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC9B,IAAI,MAAM,GAAG,MAAM,CAAC,YAAY,CAAC;QAC/B,IAAI,EAAE,OAAO,GAAG,SAAS;QACzB,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,EAAE,YAAY,EAAE,CAAC;KACjE,CAAC,CAAC;IACH,MAAM,CAAC,IAAI,CAAC,4BAA4B,EAAE,SAAS,EAAE,KAAK,CAAC,CAAC;IAC5D,MAAM,CAAC,MAAM,CAAC;AAChB,CAAC;AAOD;;;;;;;;;;;;;;;GAeG;AACH,iBAAwB,IAAW;IACjC,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC;IACjC,MAAM,CAAC,IAAI,CAAC;AACd,CAAC;AAHe,eAAO,UAGtB,CAAA;AAOD;;;;;;;;;;;;;;;GAeG;AACH,kBAAyB,KAAa;IACpC,KAAK,CAAC,YAAY,GAAG,YAAY,CAAC;IAClC,MAAM,CAAC,KAAK,CAAC;AACf,CAAC;AAHe,gBAAQ,WAGvB,CAAA;AAED;;;;;GAKG;AACH,+BAA+B,QAAgB;IAC7C,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,wBAAwB,EAAE,MAAM,CAAC,CAAC;AAC5D,CAAC;AAED;;;GAGG;AACH;IAAA;QACU,iBAAY,GAAa,EAAE,CAAC;IA2CtC,CAAC;IAzCQ,qCAAkB,GAAzB,UAA0B,QAAgB;QACxC,IAAM,kBAAkB,GAAG,IAAI,kBAAkB,EAAE,CAAC;QACpD,kBAAkB,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC;QAC1C,MAAM,CAAC,kBAAkB,CAAC;IAC5B,CAAC;IAEO,yCAAY,GAApB,UAAqB,QAAgB;QACnC,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IACpE,CAAC;IAED;;;;;;;;OAQG;IACH,kDAAqB,GAArB,UAAsB,gBAAwB;QAC5C,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,EAAE,CAAC,EAAE,CAAC;YAClD,EAAE,CAAC,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;gBAChD,MAAM,CAAC,CAAC,GAAG,CAAC,CAAC;YACf,CAAC;QACH,CAAC;QACD,MAAM,CAAC,CAAC,CAAC,CAAC;IACZ,CAAC;IAED;;;;;;OAMG;IACH,iDAAoB,GAApB,UAAqB,iBAAyB;QAC5C,IAAM,cAAc,GAAG,qBAAqB,CAAC,iBAAiB,CAAC,CAAC;QAChE,IAAM,aAAa,GAAG,IAAI,MAAM,CAAC,qBAAmB,cAAgB,CAAC,CAAC;QAEtE,MAAM,CAAC,IAAI,CAAC,qBAAqB,CAAC,aAAa,CAAC,CAAC;IACnD,CAAC;IACH,yBAAC;AAAD,CAAC,AA5CD,IA4CC;AA5CY,0BAAkB,qBA4C9B,CAAA"}