forked from OSchip/llvm-project
				
			[asan/win] Fix wrong TerminateProcess exit code
Add a test for it. llvm-svn: 286608
This commit is contained in:
		
							parent
							
								
									554fd99dd5
								
							
						
					
					
						commit
						2a2bc7293e
					
				| 
						 | 
				
			
			@ -659,7 +659,7 @@ void internal__exit(int exitcode) {
 | 
			
		|||
  // so add our own breakpoint here.
 | 
			
		||||
  if (::IsDebuggerPresent())
 | 
			
		||||
    __debugbreak();
 | 
			
		||||
  TerminateProcess(GetCurrentProcess(), 3);
 | 
			
		||||
  TerminateProcess(GetCurrentProcess(), exitcode);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uptr internal_ftruncate(fd_t fd, uptr size) {
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -0,0 +1,123 @@
 | 
			
		|||
// RUN: %clangxx_asan -g %stdcxx11 -Wno-deprecated-declarations %s -o %t
 | 
			
		||||
// RUN: %env_asan_opts=exitcode=42 %t | FileCheck %s
 | 
			
		||||
 | 
			
		||||
// CHECK: got expected 42 exit code
 | 
			
		||||
 | 
			
		||||
#include <stdlib.h>
 | 
			
		||||
#include <stdio.h>
 | 
			
		||||
 | 
			
		||||
#ifdef _WIN32
 | 
			
		||||
#include <windows.h>
 | 
			
		||||
 | 
			
		||||
int spawn_child(char **argv) {
 | 
			
		||||
  // Set an environment variable to tell the child process to interrupt
 | 
			
		||||
  // itself.
 | 
			
		||||
  if (!SetEnvironmentVariableW(L"CRASH_FOR_TEST", L"1")) {
 | 
			
		||||
    printf("SetEnvironmentVariableW failed (0x%8lx).\n", GetLastError());
 | 
			
		||||
    fflush(stdout);
 | 
			
		||||
    exit(1);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  STARTUPINFOW si;
 | 
			
		||||
  memset(&si, 0, sizeof(si));
 | 
			
		||||
  si.cb = sizeof(si);
 | 
			
		||||
 | 
			
		||||
  PROCESS_INFORMATION pi;
 | 
			
		||||
  memset(&pi, 0, sizeof(pi));
 | 
			
		||||
 | 
			
		||||
  if (!CreateProcessW(nullptr,           // No module name (use command line)
 | 
			
		||||
                      GetCommandLineW(), // Command line
 | 
			
		||||
                      nullptr,           // Process handle not inheritable
 | 
			
		||||
                      nullptr,           // Thread handle not inheritable
 | 
			
		||||
                      TRUE,              // Set handle inheritance to TRUE
 | 
			
		||||
                      0,                 // No flags
 | 
			
		||||
                      nullptr,           // Use parent's environment block
 | 
			
		||||
                      nullptr,           // Use parent's starting directory
 | 
			
		||||
                      &si, &pi)) {
 | 
			
		||||
    printf("CreateProcess failed (0x%08lx).\n", GetLastError());
 | 
			
		||||
    fflush(stdout);
 | 
			
		||||
    exit(1);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  WaitForSingleObject(pi.hProcess, INFINITE);
 | 
			
		||||
 | 
			
		||||
  DWORD exit_code;
 | 
			
		||||
  if (!GetExitCodeProcess(pi.hProcess, &exit_code)) {
 | 
			
		||||
    printf("GetExitCodeProcess failed (0x%08lx).\n", GetLastError());
 | 
			
		||||
    fflush(stdout);
 | 
			
		||||
    exit(1);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  CloseHandle(pi.hProcess);
 | 
			
		||||
  CloseHandle(pi.hThread);
 | 
			
		||||
 | 
			
		||||
  return exit_code;
 | 
			
		||||
}
 | 
			
		||||
#else
 | 
			
		||||
#include <spawn.h>
 | 
			
		||||
#include <errno.h>
 | 
			
		||||
#include <sys/wait.h>
 | 
			
		||||
 | 
			
		||||
#if defined(__APPLE__) && !(defined(TARGET_OS_IPHONE) && TARGET_OS_IPHONE)
 | 
			
		||||
#define USE_NSGETENVIRON 1
 | 
			
		||||
#else
 | 
			
		||||
#define USE_NSGETENVIRON 0
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#if !USE_NSGETENVIRON
 | 
			
		||||
extern char **environ;
 | 
			
		||||
#else
 | 
			
		||||
#include <crt_externs.h> // _NSGetEnviron
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
int spawn_child(char **argv) {
 | 
			
		||||
  setenv("CRASH_FOR_TEST", "1", 1);
 | 
			
		||||
 | 
			
		||||
#if !USE_NSGETENVIRON
 | 
			
		||||
  char **envp = environ;
 | 
			
		||||
#else
 | 
			
		||||
  char **envp = *_NSGetEnviron();
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
  pid_t pid;
 | 
			
		||||
  int err = posix_spawn(&pid, argv[0], nullptr, nullptr, argv, envp);
 | 
			
		||||
  if (err) {
 | 
			
		||||
    printf("posix_spawn failed: %d\n", err);
 | 
			
		||||
    fflush(stdout);
 | 
			
		||||
    exit(1);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Wait until the child exits.
 | 
			
		||||
  int status;
 | 
			
		||||
  pid_t wait_result_pid;
 | 
			
		||||
  do {
 | 
			
		||||
    wait_result_pid = waitpid(pid, &status, 0);
 | 
			
		||||
  } while (wait_result_pid == -1 && errno == EINTR);
 | 
			
		||||
 | 
			
		||||
  if (wait_result_pid != pid || !WIFEXITED(status)) {
 | 
			
		||||
    printf("error in waitpid\n");
 | 
			
		||||
    fflush(stdout);
 | 
			
		||||
    exit(1);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // Return the exit status.
 | 
			
		||||
  return WEXITSTATUS(status);
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
int main(int argc, char **argv) {
 | 
			
		||||
  int r = 0;
 | 
			
		||||
  if (getenv("CRASH_FOR_TEST")) {
 | 
			
		||||
    // Generate an asan report to test ASAN_OPTIONS=exitcode=42
 | 
			
		||||
    int *p = new int;
 | 
			
		||||
    delete p;
 | 
			
		||||
    r = *p;
 | 
			
		||||
  } else {
 | 
			
		||||
    int exit_code = spawn_child(argv);
 | 
			
		||||
    if (exit_code == 42) {
 | 
			
		||||
      printf("got expected 42 exit code\n");
 | 
			
		||||
      fflush(stdout);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  return r;
 | 
			
		||||
}
 | 
			
		||||
		Loading…
	
		Reference in New Issue