Function walkStackWindows [src]

Prototype

pub noinline fn walkStackWindows(addresses: []usize, existing_context: ?*const windows.CONTEXT) usize

Parameters

addresses: []usizeexisting_context: ?*const windows.CONTEXT

Source

pub noinline fn walkStackWindows(addresses: []usize, existing_context: ?*const windows.CONTEXT) usize { if (builtin.cpu.arch == .x86) { // RtlVirtualUnwind doesn't exist on x86 return windows.ntdll.RtlCaptureStackBackTrace(0, addresses.len, @as(**anyopaque, @ptrCast(addresses.ptr)), null); } const tib = &windows.teb().NtTib; var context: windows.CONTEXT = undefined; if (existing_context) |context_ptr| { context = context_ptr.*; } else { context = std.mem.zeroes(windows.CONTEXT); windows.ntdll.RtlCaptureContext(&context); } var i: usize = 0; var image_base: windows.DWORD64 = undefined; var history_table: windows.UNWIND_HISTORY_TABLE = std.mem.zeroes(windows.UNWIND_HISTORY_TABLE); while (i < addresses.len) : (i += 1) { const current_regs = context.getRegs(); if (windows.ntdll.RtlLookupFunctionEntry(current_regs.ip, &image_base, &history_table)) |runtime_function| { var handler_data: ?*anyopaque = null; var establisher_frame: u64 = undefined; _ = windows.ntdll.RtlVirtualUnwind( windows.UNW_FLAG_NHANDLER, image_base, current_regs.ip, runtime_function, &context, &handler_data, &establisher_frame, null, ); } else { // leaf function context.setIp(@as(*usize, @ptrFromInt(current_regs.sp)).*); context.setSp(current_regs.sp + @sizeOf(usize)); } const next_regs = context.getRegs(); if (next_regs.sp < @intFromPtr(tib.StackLimit) or next_regs.sp > @intFromPtr(tib.StackBase)) { break; } if (next_regs.ip == 0) { break; } addresses[i] = next_regs.ip; } return i; }