112792af0SZachary Turner //===-- ConnectionGenericFileWindows.cpp ------------------------*- C++ -*-===//
212792af0SZachary Turner //
312792af0SZachary Turner //                     The LLVM Compiler Infrastructure
412792af0SZachary Turner //
512792af0SZachary Turner // This file is distributed under the University of Illinois Open Source
612792af0SZachary Turner // License. See LICENSE.TXT for details.
712792af0SZachary Turner //
812792af0SZachary Turner //===----------------------------------------------------------------------===//
912792af0SZachary Turner 
1012792af0SZachary Turner #include "lldb/Core/Error.h"
1112792af0SZachary Turner #include "lldb/Core/Log.h"
1212792af0SZachary Turner #include "lldb/Host/TimeValue.h"
1312792af0SZachary Turner #include "lldb/Host/windows/ConnectionGenericFileWindows.h"
1412792af0SZachary Turner 
1512792af0SZachary Turner #include "llvm/ADT/STLExtras.h"
1612792af0SZachary Turner #include "llvm/ADT/StringRef.h"
17*190fadcdSZachary Turner #include "llvm/Support/ConvertUTF.h"
1812792af0SZachary Turner 
1912792af0SZachary Turner using namespace lldb;
2012792af0SZachary Turner using namespace lldb_private;
2112792af0SZachary Turner 
2212792af0SZachary Turner namespace
2312792af0SZachary Turner {
2412792af0SZachary Turner // This is a simple helper class to package up the information needed to return from a Read/Write
2512792af0SZachary Turner // operation function.  Since there is a lot of code to be run before exit regardless of whether the
2612792af0SZachary Turner // operation succeeded or failed, combined with many possible return paths, this is the cleanest
2712792af0SZachary Turner // way to represent it.
2812792af0SZachary Turner class ReturnInfo
2912792af0SZachary Turner {
3012792af0SZachary Turner   public:
3112792af0SZachary Turner     void
3212792af0SZachary Turner     Set(size_t bytes, ConnectionStatus status, DWORD error_code)
3312792af0SZachary Turner     {
3412792af0SZachary Turner         m_error.SetError(error_code, eErrorTypeWin32);
3512792af0SZachary Turner         m_bytes = bytes;
3612792af0SZachary Turner         m_status = status;
3712792af0SZachary Turner     }
3812792af0SZachary Turner 
3912792af0SZachary Turner     void
4012792af0SZachary Turner     Set(size_t bytes, ConnectionStatus status, llvm::StringRef error_msg)
4112792af0SZachary Turner     {
4212792af0SZachary Turner         m_error.SetErrorString(error_msg.data());
4312792af0SZachary Turner         m_bytes = bytes;
4412792af0SZachary Turner         m_status = status;
4512792af0SZachary Turner     }
4612792af0SZachary Turner 
4712792af0SZachary Turner     size_t
4812792af0SZachary Turner     GetBytes() const
4912792af0SZachary Turner     {
5012792af0SZachary Turner         return m_bytes;
5112792af0SZachary Turner     }
5212792af0SZachary Turner     ConnectionStatus
5312792af0SZachary Turner     GetStatus() const
5412792af0SZachary Turner     {
5512792af0SZachary Turner         return m_status;
5612792af0SZachary Turner     }
5712792af0SZachary Turner     const Error &
5812792af0SZachary Turner     GetError() const
5912792af0SZachary Turner     {
6012792af0SZachary Turner         return m_error;
6112792af0SZachary Turner     }
6212792af0SZachary Turner 
6312792af0SZachary Turner   private:
6412792af0SZachary Turner     Error m_error;
6512792af0SZachary Turner     size_t m_bytes;
6612792af0SZachary Turner     ConnectionStatus m_status;
6712792af0SZachary Turner };
6812792af0SZachary Turner }
6912792af0SZachary Turner 
7012792af0SZachary Turner ConnectionGenericFile::ConnectionGenericFile()
7112792af0SZachary Turner     : m_file(INVALID_HANDLE_VALUE)
7212792af0SZachary Turner     , m_owns_file(false)
7312792af0SZachary Turner {
7412792af0SZachary Turner     ::ZeroMemory(&m_overlapped, sizeof(m_overlapped));
7512792af0SZachary Turner     ::ZeroMemory(&m_file_position, sizeof(m_file_position));
7612792af0SZachary Turner     InitializeEventHandles();
7712792af0SZachary Turner }
7812792af0SZachary Turner 
7912792af0SZachary Turner ConnectionGenericFile::ConnectionGenericFile(lldb::file_t file, bool owns_file)
8012792af0SZachary Turner     : m_file(file)
8112792af0SZachary Turner     , m_owns_file(owns_file)
8212792af0SZachary Turner {
8312792af0SZachary Turner     ::ZeroMemory(&m_overlapped, sizeof(m_overlapped));
8412792af0SZachary Turner     ::ZeroMemory(&m_file_position, sizeof(m_file_position));
8512792af0SZachary Turner     InitializeEventHandles();
8612792af0SZachary Turner }
8712792af0SZachary Turner 
8812792af0SZachary Turner ConnectionGenericFile::~ConnectionGenericFile()
8912792af0SZachary Turner {
9012792af0SZachary Turner     if (m_owns_file && IsConnected())
9112792af0SZachary Turner         ::CloseHandle(m_file);
9212792af0SZachary Turner 
9312792af0SZachary Turner     ::CloseHandle(m_event_handles[kBytesAvailableEvent]);
9412792af0SZachary Turner     ::CloseHandle(m_event_handles[kInterruptEvent]);
9512792af0SZachary Turner }
9612792af0SZachary Turner 
9712792af0SZachary Turner void
9812792af0SZachary Turner ConnectionGenericFile::InitializeEventHandles()
9912792af0SZachary Turner {
10012792af0SZachary Turner     m_event_handles[kInterruptEvent] = CreateEvent(NULL, FALSE, FALSE, NULL);
10112792af0SZachary Turner 
10212792af0SZachary Turner     // Note, we should use a manual reset event for the hEvent argument of the OVERLAPPED.  This
10312792af0SZachary Turner     // is because both WaitForMultipleObjects and GetOverlappedResult (if you set the bWait
10412792af0SZachary Turner     // argument to TRUE) will wait for the event to be signalled.  If we use an auto-reset event,
10512792af0SZachary Turner     // WaitForMultipleObjects will reset the event, return successfully, and then
10612792af0SZachary Turner     // GetOverlappedResult will block since the event is no longer signalled.
10712792af0SZachary Turner     m_event_handles[kBytesAvailableEvent] = ::CreateEvent(NULL, TRUE, FALSE, NULL);
10812792af0SZachary Turner }
10912792af0SZachary Turner 
11012792af0SZachary Turner bool
11112792af0SZachary Turner ConnectionGenericFile::IsConnected() const
11212792af0SZachary Turner {
11312792af0SZachary Turner     return m_file && (m_file != INVALID_HANDLE_VALUE);
11412792af0SZachary Turner }
11512792af0SZachary Turner 
11612792af0SZachary Turner lldb::ConnectionStatus
11712792af0SZachary Turner ConnectionGenericFile::Connect(const char *s, Error *error_ptr)
11812792af0SZachary Turner {
11912792af0SZachary Turner     Log *log(lldb_private::GetLogIfAnyCategoriesSet(LIBLLDB_LOG_CONNECTION));
12012792af0SZachary Turner     if (log)
12112792af0SZachary Turner         log->Printf("%p ConnectionGenericFile::Connect (url = '%s')", static_cast<void *>(this), s);
12212792af0SZachary Turner 
12312792af0SZachary Turner     if (strstr(s, "file://") != s)
12412792af0SZachary Turner     {
12512792af0SZachary Turner         if (error_ptr)
12612792af0SZachary Turner             error_ptr->SetErrorStringWithFormat("unsupported connection URL: '%s'", s);
12712792af0SZachary Turner         return eConnectionStatusError;
12812792af0SZachary Turner     }
12912792af0SZachary Turner 
13012792af0SZachary Turner     if (IsConnected())
13112792af0SZachary Turner     {
13212792af0SZachary Turner         ConnectionStatus status = Disconnect(error_ptr);
13312792af0SZachary Turner         if (status != eConnectionStatusSuccess)
13412792af0SZachary Turner             return status;
13512792af0SZachary Turner     }
13612792af0SZachary Turner 
13712792af0SZachary Turner     // file://PATH
13812792af0SZachary Turner     const char *path = s + strlen("file://");
13912792af0SZachary Turner     // Open the file for overlapped access.  If it does not exist, create it.  We open it overlapped
14012792af0SZachary Turner     // so that we can issue asynchronous reads and then use WaitForMultipleObjects to allow the read
14112792af0SZachary Turner     // to be interrupted by an event object.
142*190fadcdSZachary Turner     std::wstring wpath;
143*190fadcdSZachary Turner     if (!llvm::ConvertUTF8toWide(path, wpath))
144*190fadcdSZachary Turner     {
145*190fadcdSZachary Turner         if (error_ptr)
146*190fadcdSZachary Turner             error_ptr->SetError(1, eErrorTypeGeneric);
147*190fadcdSZachary Turner         return eConnectionStatusError;
148*190fadcdSZachary Turner     }
149*190fadcdSZachary Turner     m_file = ::CreateFileW(wpath.c_str(), GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ, NULL, OPEN_ALWAYS,
150*190fadcdSZachary Turner                            FILE_FLAG_OVERLAPPED, NULL);
15112792af0SZachary Turner     if (m_file == INVALID_HANDLE_VALUE)
15212792af0SZachary Turner     {
15312792af0SZachary Turner         if (error_ptr)
15412792af0SZachary Turner             error_ptr->SetError(::GetLastError(), eErrorTypeWin32);
15512792af0SZachary Turner         return eConnectionStatusError;
15612792af0SZachary Turner     }
15712792af0SZachary Turner 
15812792af0SZachary Turner     m_owns_file = true;
159eb303ee5SVince Harron     m_uri.assign(s);
16012792af0SZachary Turner     return eConnectionStatusSuccess;
16112792af0SZachary Turner }
16212792af0SZachary Turner 
16312792af0SZachary Turner lldb::ConnectionStatus
16412792af0SZachary Turner ConnectionGenericFile::Disconnect(Error *error_ptr)
16512792af0SZachary Turner {
16612792af0SZachary Turner     Log *log(lldb_private::GetLogIfAnyCategoriesSet(LIBLLDB_LOG_CONNECTION));
16712792af0SZachary Turner     if (log)
1685cbeb850SZachary Turner         log->Printf("%p ConnectionGenericFile::Disconnect ()", static_cast<void *>(this));
16912792af0SZachary Turner 
17012792af0SZachary Turner     if (!IsConnected())
17112792af0SZachary Turner         return eConnectionStatusSuccess;
17212792af0SZachary Turner 
17312792af0SZachary Turner     // Reset the handle so that after we unblock any pending reads, subsequent calls to Read() will
17412792af0SZachary Turner     // see a disconnected state.
17512792af0SZachary Turner     HANDLE old_file = m_file;
17612792af0SZachary Turner     m_file = INVALID_HANDLE_VALUE;
17712792af0SZachary Turner 
17812792af0SZachary Turner     // Set the disconnect event so that any blocking reads unblock, then cancel any pending IO operations.
17912792af0SZachary Turner     ::CancelIoEx(old_file, &m_overlapped);
18012792af0SZachary Turner 
18112792af0SZachary Turner     // Close the file handle if we owned it, but don't close the event handles.  We could always
18212792af0SZachary Turner     // reconnect with the same Connection instance.
18312792af0SZachary Turner     if (m_owns_file)
18412792af0SZachary Turner         ::CloseHandle(old_file);
18512792af0SZachary Turner 
18612792af0SZachary Turner     ::ZeroMemory(&m_file_position, sizeof(m_file_position));
18712792af0SZachary Turner     m_owns_file = false;
188eb303ee5SVince Harron     m_uri.clear();
18912792af0SZachary Turner     return eConnectionStatusSuccess;
19012792af0SZachary Turner }
19112792af0SZachary Turner 
19212792af0SZachary Turner size_t
19312792af0SZachary Turner ConnectionGenericFile::Read(void *dst, size_t dst_len, uint32_t timeout_usec, lldb::ConnectionStatus &status, Error *error_ptr)
19412792af0SZachary Turner {
19512792af0SZachary Turner     ReturnInfo return_info;
196ae944608SHafiz Abid Qadeer     BOOL result = 0;
197ae944608SHafiz Abid Qadeer     DWORD bytes_read = 0;
19812792af0SZachary Turner 
19912792af0SZachary Turner     if (error_ptr)
20012792af0SZachary Turner         error_ptr->Clear();
20112792af0SZachary Turner 
20212792af0SZachary Turner     if (!IsConnected())
20312792af0SZachary Turner     {
20412792af0SZachary Turner         return_info.Set(0, eConnectionStatusNoConnection, ERROR_INVALID_HANDLE);
20512792af0SZachary Turner         goto finish;
20612792af0SZachary Turner     }
20712792af0SZachary Turner 
20812792af0SZachary Turner     m_overlapped.hEvent = m_event_handles[kBytesAvailableEvent];
20912792af0SZachary Turner 
210ae944608SHafiz Abid Qadeer     result = ::ReadFile(m_file, dst, dst_len, NULL, &m_overlapped);
21112792af0SZachary Turner     if (result || ::GetLastError() == ERROR_IO_PENDING)
21212792af0SZachary Turner     {
21312792af0SZachary Turner         if (!result)
21412792af0SZachary Turner         {
21512792af0SZachary Turner             // The expected return path.  The operation is pending.  Wait for the operation to complete
21612792af0SZachary Turner             // or be interrupted.
21712792af0SZachary Turner             TimeValue time_value;
21812792af0SZachary Turner             time_value.OffsetWithMicroSeconds(timeout_usec);
21912792af0SZachary Turner             DWORD milliseconds = time_value.milliseconds();
22048b475cbSZachary Turner             DWORD wait_result = ::WaitForMultipleObjects(llvm::array_lengthof(m_event_handles), m_event_handles, FALSE, milliseconds);
22112792af0SZachary Turner             // All of the events are manual reset events, so make sure we reset them to non-signalled.
22248b475cbSZachary Turner             switch (wait_result)
22312792af0SZachary Turner             {
22412792af0SZachary Turner                 case WAIT_OBJECT_0 + kBytesAvailableEvent:
22512792af0SZachary Turner                     break;
22612792af0SZachary Turner                 case WAIT_OBJECT_0 + kInterruptEvent:
22712792af0SZachary Turner                     return_info.Set(0, eConnectionStatusInterrupted, 0);
22812792af0SZachary Turner                     goto finish;
22912792af0SZachary Turner                 case WAIT_TIMEOUT:
23012792af0SZachary Turner                     return_info.Set(0, eConnectionStatusTimedOut, 0);
23112792af0SZachary Turner                     goto finish;
23212792af0SZachary Turner                 case WAIT_FAILED:
23312792af0SZachary Turner                     return_info.Set(0, eConnectionStatusError, ::GetLastError());
23412792af0SZachary Turner                     goto finish;
23512792af0SZachary Turner             }
23612792af0SZachary Turner         }
23712792af0SZachary Turner         // The data is ready.  Figure out how much was read and return;
23812792af0SZachary Turner         if (!::GetOverlappedResult(m_file, &m_overlapped, &bytes_read, FALSE))
23912792af0SZachary Turner         {
24012792af0SZachary Turner             DWORD result_error = ::GetLastError();
24112792af0SZachary Turner             // ERROR_OPERATION_ABORTED occurs when someone calls Disconnect() during a blocking read.
24212792af0SZachary Turner             // This triggers a call to CancelIoEx, which causes the operation to complete and the
24312792af0SZachary Turner             // result to be ERROR_OPERATION_ABORTED.
2448ada0b97SZachary Turner             if (result_error == ERROR_HANDLE_EOF || result_error == ERROR_OPERATION_ABORTED || result_error == ERROR_BROKEN_PIPE)
24512792af0SZachary Turner                 return_info.Set(bytes_read, eConnectionStatusEndOfFile, 0);
24612792af0SZachary Turner             else
24712792af0SZachary Turner                 return_info.Set(bytes_read, eConnectionStatusError, result_error);
24812792af0SZachary Turner         }
24912792af0SZachary Turner         else if (bytes_read == 0)
25012792af0SZachary Turner             return_info.Set(bytes_read, eConnectionStatusEndOfFile, 0);
25112792af0SZachary Turner         else
25212792af0SZachary Turner             return_info.Set(bytes_read, eConnectionStatusSuccess, 0);
25312792af0SZachary Turner 
25412792af0SZachary Turner         goto finish;
25512792af0SZachary Turner     }
256b2df30d6SZachary Turner     else if (::GetLastError() == ERROR_BROKEN_PIPE)
257b2df30d6SZachary Turner     {
258b2df30d6SZachary Turner         // The write end of a pipe was closed.  This is equivalent to EOF.
259b2df30d6SZachary Turner         return_info.Set(0, eConnectionStatusEndOfFile, 0);
260b2df30d6SZachary Turner     }
261b2df30d6SZachary Turner     else
262b2df30d6SZachary Turner     {
263e171da5cSBruce Mitchener         // An unknown error occurred.  Fail out.
26412792af0SZachary Turner         return_info.Set(0, eConnectionStatusError, ::GetLastError());
265b2df30d6SZachary Turner     }
26612792af0SZachary Turner     goto finish;
26712792af0SZachary Turner 
26812792af0SZachary Turner finish:
26912792af0SZachary Turner     status = return_info.GetStatus();
27012792af0SZachary Turner     if (error_ptr)
27112792af0SZachary Turner         *error_ptr = return_info.GetError();
27212792af0SZachary Turner 
27312792af0SZachary Turner     // kBytesAvailableEvent is a manual reset event.  Make sure it gets reset here so that any
27412792af0SZachary Turner     // subsequent operations don't immediately see bytes available.
27512792af0SZachary Turner     ResetEvent(m_event_handles[kBytesAvailableEvent]);
27612792af0SZachary Turner 
27712792af0SZachary Turner     IncrementFilePointer(return_info.GetBytes());
27812792af0SZachary Turner     Log *log(lldb_private::GetLogIfAnyCategoriesSet(LIBLLDB_LOG_CONNECTION));
27912792af0SZachary Turner     if (log)
28012792af0SZachary Turner     {
28112792af0SZachary Turner         log->Printf("%" PRIxPTR " ConnectionGenericFile::Read()  handle = %" PRIxPTR ", dst = %" PRIxPTR ", dst_len = %" PRIu64
28212792af0SZachary Turner                     ") => %" PRIu64 ", error = %s",
28312792af0SZachary Turner                     this, m_file, dst, static_cast<uint64_t>(dst_len), static_cast<uint64_t>(return_info.GetBytes()),
28412792af0SZachary Turner                     return_info.GetError().AsCString());
28512792af0SZachary Turner     }
28612792af0SZachary Turner 
28712792af0SZachary Turner     return return_info.GetBytes();
28812792af0SZachary Turner }
28912792af0SZachary Turner 
29012792af0SZachary Turner size_t
29112792af0SZachary Turner ConnectionGenericFile::Write(const void *src, size_t src_len, lldb::ConnectionStatus &status, Error *error_ptr)
29212792af0SZachary Turner {
29312792af0SZachary Turner     ReturnInfo return_info;
294ae944608SHafiz Abid Qadeer     DWORD bytes_written = 0;
295ae944608SHafiz Abid Qadeer     BOOL result = 0;
29612792af0SZachary Turner 
29712792af0SZachary Turner     if (error_ptr)
29812792af0SZachary Turner         error_ptr->Clear();
29912792af0SZachary Turner 
30012792af0SZachary Turner     if (!IsConnected())
30112792af0SZachary Turner     {
30212792af0SZachary Turner         return_info.Set(0, eConnectionStatusNoConnection, ERROR_INVALID_HANDLE);
30312792af0SZachary Turner         goto finish;
30412792af0SZachary Turner     }
30512792af0SZachary Turner 
30612792af0SZachary Turner     m_overlapped.hEvent = NULL;
30712792af0SZachary Turner 
30812792af0SZachary Turner     // Writes are not interruptible like reads are, so just block until it's done.
309ae944608SHafiz Abid Qadeer     result = ::WriteFile(m_file, src, src_len, NULL, &m_overlapped);
31012792af0SZachary Turner     if (!result && ::GetLastError() != ERROR_IO_PENDING)
31112792af0SZachary Turner     {
31212792af0SZachary Turner         return_info.Set(0, eConnectionStatusError, ::GetLastError());
31312792af0SZachary Turner         goto finish;
31412792af0SZachary Turner     }
31512792af0SZachary Turner 
31612792af0SZachary Turner     if (!::GetOverlappedResult(m_file, &m_overlapped, &bytes_written, TRUE))
31712792af0SZachary Turner     {
31812792af0SZachary Turner         return_info.Set(bytes_written, eConnectionStatusError, ::GetLastError());
31912792af0SZachary Turner         goto finish;
32012792af0SZachary Turner     }
32112792af0SZachary Turner 
32212792af0SZachary Turner     return_info.Set(bytes_written, eConnectionStatusSuccess, 0);
32312792af0SZachary Turner     goto finish;
32412792af0SZachary Turner 
32512792af0SZachary Turner finish:
32612792af0SZachary Turner     status = return_info.GetStatus();
32712792af0SZachary Turner     if (error_ptr)
32812792af0SZachary Turner         *error_ptr = return_info.GetError();
32912792af0SZachary Turner 
33012792af0SZachary Turner     IncrementFilePointer(return_info.GetBytes());
33112792af0SZachary Turner     Log *log(lldb_private::GetLogIfAnyCategoriesSet(LIBLLDB_LOG_CONNECTION));
33212792af0SZachary Turner     if (log)
33312792af0SZachary Turner     {
33412792af0SZachary Turner         log->Printf("%" PRIxPTR " ConnectionGenericFile::Write()  handle = %" PRIxPTR ", src = %" PRIxPTR ", src_len = %" PRIu64
33512792af0SZachary Turner                     ") => %" PRIu64 ", error = %s",
33612792af0SZachary Turner                     this, m_file, src, static_cast<uint64_t>(src_len), static_cast<uint64_t>(return_info.GetBytes()),
33712792af0SZachary Turner                     return_info.GetError().AsCString());
33812792af0SZachary Turner     }
33912792af0SZachary Turner     return return_info.GetBytes();
34012792af0SZachary Turner }
34112792af0SZachary Turner 
342eb303ee5SVince Harron std::string
343eb303ee5SVince Harron ConnectionGenericFile::GetURI()
344eb303ee5SVince Harron {
345eb303ee5SVince Harron     return m_uri;
346eb303ee5SVince Harron }
347eb303ee5SVince Harron 
34812792af0SZachary Turner bool
34912792af0SZachary Turner ConnectionGenericFile::InterruptRead()
35012792af0SZachary Turner {
35112792af0SZachary Turner     return ::SetEvent(m_event_handles[kInterruptEvent]);
35212792af0SZachary Turner }
35312792af0SZachary Turner 
35412792af0SZachary Turner void
35512792af0SZachary Turner ConnectionGenericFile::IncrementFilePointer(DWORD amount)
35612792af0SZachary Turner {
35712792af0SZachary Turner     LARGE_INTEGER old_pos;
35812792af0SZachary Turner     old_pos.HighPart = m_overlapped.OffsetHigh;
35912792af0SZachary Turner     old_pos.LowPart = m_overlapped.Offset;
36012792af0SZachary Turner     old_pos.QuadPart += amount;
36112792af0SZachary Turner     m_overlapped.Offset = old_pos.LowPart;
36212792af0SZachary Turner     m_overlapped.OffsetHigh = old_pos.HighPart;
36312792af0SZachary Turner }
364