-
Notifications
You must be signed in to change notification settings - Fork 3.8k
[Support] Add Interrupt Handling in Pipe #16234
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,77 @@ | ||
| /* | ||
| * Licensed to the Apache Software Foundation (ASF) under one | ||
| * or more contributor license agreements. See the NOTICE file | ||
| * distributed with this work for additional information | ||
| * regarding copyright ownership. The ASF licenses this file | ||
| * to you under the Apache License, Version 2.0 (the | ||
| * "License"); you may not use this file except in compliance | ||
| * with the License. You may obtain a copy of the License at | ||
| * | ||
| * http://www.apache.org/licenses/LICENSE-2.0 | ||
| * | ||
| * Unless required by applicable law or agreed to in writing, | ||
| * software distributed under the License is distributed on an | ||
| * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY | ||
| * KIND, either express or implied. See the License for the | ||
| * specific language governing permissions and limitations | ||
| * under the License. | ||
| */ | ||
|
|
||
| /*! | ||
| * \file err_handling.h | ||
| * \brief Common error handling functions for socket.h and pipe.h | ||
| */ | ||
| #ifndef TVM_SUPPORT_ERR_HANDLING_H_ | ||
| #define TVM_SUPPORT_ERR_HANDLING_H_ | ||
| #include <errno.h> | ||
|
|
||
| namespace tvm { | ||
| namespace support { | ||
| /*! | ||
| * \return last error of socket operation | ||
| */ | ||
| static int GetLastErrorCode() { | ||
|
Member
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This needs to be passed as a second template function parameter to RetryCallOnEINTR
Member
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We need specialized function in socket and pipe For socket it should be |
||
| #ifdef _WIN32 | ||
| return WSAGetLastError(); | ||
| #else | ||
| return errno; | ||
| #endif | ||
| } | ||
| /*! | ||
| * \brief Call a function and retry if an EINTR error is encountered. | ||
| * | ||
| * Socket operations can return EINTR when the interrupt handler | ||
| * is registered by the execution environment(e.g. python). | ||
| * We should retry if there is no KeyboardInterrupt recorded in | ||
| * the environment. | ||
| * | ||
| * \note This function is needed to avoid rare interrupt event | ||
| * in long running server code. | ||
| * | ||
| * \param func The function to retry. | ||
| * \return The return code returned by function f or error_value on retry failure. | ||
| */ | ||
| template <typename FuncType> | ||
| inline ssize_t RetryCallOnEINTR(FuncType func) { | ||
| ssize_t ret = func(); | ||
| // common path | ||
| if (ret != -1) return ret; | ||
| // less common path | ||
| do { | ||
| if (GetLastErrorCode() == EINTR) { | ||
| // Call into env check signals to see if there are | ||
| // environment specific(e.g. python) signal exceptions. | ||
| // This function will throw an exception if there is | ||
| // if the process received a signal that requires TVM to return immediately (e.g. SIGINT). | ||
| runtime::EnvCheckSignals(); | ||
| } else { | ||
| // other errors | ||
| return ret; | ||
| } | ||
| ret = func(); | ||
| } while (ret == -1); | ||
| return ret; | ||
| } | ||
| } // namespace support | ||
| } // namespace tvm | ||
| #endif // TVM_SUPPORT_ERR_HANDLING_H_ | ||
| Original file line number | Diff line number | Diff line change |
|---|---|---|
|
|
@@ -36,6 +36,7 @@ | |
| #include <cstdlib> | ||
| #include <cstring> | ||
| #endif | ||
| #include "error_handling.h" | ||
|
|
||
| namespace tvm { | ||
| namespace support { | ||
|
|
@@ -52,6 +53,7 @@ class Pipe : public dmlc::Stream { | |
| #endif | ||
| /*! \brief destructor */ | ||
| ~Pipe() { Flush(); } | ||
|
|
||
| using Stream::Read; | ||
| using Stream::Write; | ||
| /*! | ||
|
|
@@ -67,8 +69,8 @@ class Pipe : public dmlc::Stream { | |
| ICHECK(ReadFile(handle_, static_cast<TCHAR*>(ptr), size, &nread, nullptr)) | ||
|
Member
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. on windows, create a lambda function that follows the same convention auto fread = [&]() {
if (!ReadFile(handle_, static_cast<TCHAR*>(ptr), size, &nread, nullptr)) return -1;
if (nread != nwrite) return -1;
return nread;
};
|
||
| << "Read Error: " << GetLastError(); | ||
| #else | ||
| ssize_t nread; | ||
| nread = read(handle_, ptr, size); | ||
| ssize_t nread = RetryCallOnEINTR( | ||
| [&]() { return read(handle_, ptr, size); }); | ||
| ICHECK_GE(nread, 0) << "Write Error: " << strerror(errno); | ||
| #endif | ||
| return static_cast<size_t>(nread); | ||
|
|
@@ -87,8 +89,8 @@ class Pipe : public dmlc::Stream { | |
| static_cast<size_t>(nwrite) == size) | ||
| << "Write Error: " << GetLastError(); | ||
| #else | ||
| ssize_t nwrite; | ||
| nwrite = write(handle_, ptr, size); | ||
| ssize_t nwrite = RetryCallOnEINTR( | ||
| [&]() { return write(handle_, ptr, size); }); | ||
| ICHECK_EQ(static_cast<size_t>(nwrite), size) << "Write Error: " << strerror(errno); | ||
| #endif | ||
| } | ||
|
|
||
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
errno_handling (specific to handle error number)