| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |     An implementation of Windows console I/O | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Classes defined here: _WindowsConsoleIO | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Written by Steve Dower | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define PY_SSIZE_T_CLEAN
 | 
					
						
							|  |  |  | #include "Python.h"
 | 
					
						
							| 
									
										
										
										
											2021-10-13 15:03:35 +02:00
										 |  |  | #include "pycore_fileutils.h"     // _Py_BEGIN_SUPPRESS_IPH
 | 
					
						
							|  |  |  | #include "pycore_object.h"        // _PyObject_GC_UNTRACK()
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef MS_WINDOWS
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-15 02:35:41 +02:00
										 |  |  | #include "structmember.h"         // PyMemberDef
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | #ifdef HAVE_SYS_TYPES_H
 | 
					
						
							|  |  |  | #include <sys/types.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #ifdef HAVE_SYS_STAT_H
 | 
					
						
							|  |  |  | #include <sys/stat.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #include <stddef.h> /* For offsetof */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define WIN32_LEAN_AND_MEAN
 | 
					
						
							|  |  |  | #include <windows.h>
 | 
					
						
							| 
									
										
										
										
											2016-09-17 13:51:23 -07:00
										 |  |  | #include <fcntl.h>
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "_iomodule.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* BUFSIZ determines how many characters can be typed at the console
 | 
					
						
							|  |  |  |    before it starts blocking. */ | 
					
						
							|  |  |  | #if BUFSIZ < (16*1024)
 | 
					
						
							|  |  |  | #define SMALLCHUNK (2*1024)
 | 
					
						
							|  |  |  | #elif (BUFSIZ >= (2 << 25))
 | 
					
						
							| 
									
										
										
										
											2017-11-08 14:44:44 -08:00
										 |  |  | #error "unreasonable BUFSIZ > 64 MiB defined"
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | #else
 | 
					
						
							|  |  |  | #define SMALLCHUNK BUFSIZ
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* BUFMAX determines how many bytes can be read in one go. */ | 
					
						
							|  |  |  | #define BUFMAX (32*1024*1024)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-03 09:04:58 -07:00
										 |  |  | /* SMALLBUF determines how many utf-8 characters will be
 | 
					
						
							|  |  |  |    buffered within the stream, in order to support reads | 
					
						
							|  |  |  |    of less than one character */ | 
					
						
							|  |  |  | #define SMALLBUF 4
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | char _get_console_type(HANDLE handle) { | 
					
						
							|  |  |  |     DWORD mode, peek_count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (handle == INVALID_HANDLE_VALUE) | 
					
						
							|  |  |  |         return '\0'; | 
					
						
							| 
									
										
										
										
											2016-09-19 22:17:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     if (!GetConsoleMode(handle, &mode)) | 
					
						
							|  |  |  |         return '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Peek at the handle to see whether it is an input or output handle */ | 
					
						
							|  |  |  |     if (GetNumberOfConsoleInputEvents(handle, &peek_count)) | 
					
						
							|  |  |  |         return 'r'; | 
					
						
							|  |  |  |     return 'w'; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char _PyIO_get_console_type(PyObject *path_or_fd) { | 
					
						
							| 
									
										
										
										
											2017-02-04 15:07:46 -08:00
										 |  |  |     int fd = PyLong_AsLong(path_or_fd); | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     PyErr_Clear(); | 
					
						
							|  |  |  |     if (fd >= 0) { | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |         HANDLE handle = _Py_get_osfhandle_noraise(fd); | 
					
						
							| 
									
										
										
										
											2017-02-04 15:07:46 -08:00
										 |  |  |         if (handle == INVALID_HANDLE_VALUE) | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |             return '\0'; | 
					
						
							|  |  |  |         return _get_console_type(handle); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-04 15:07:46 -08:00
										 |  |  |     PyObject *decoded; | 
					
						
							|  |  |  |     wchar_t *decoded_wstr; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-04 15:07:46 -08:00
										 |  |  |     if (!PyUnicode_FSDecoder(path_or_fd, &decoded)) { | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         PyErr_Clear(); | 
					
						
							| 
									
										
										
										
											2016-09-17 13:51:23 -07:00
										 |  |  |         return '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-02-04 15:07:46 -08:00
										 |  |  |     decoded_wstr = PyUnicode_AsWideCharString(decoded, NULL); | 
					
						
							| 
									
										
										
										
											2016-09-17 13:51:23 -07:00
										 |  |  |     Py_CLEAR(decoded); | 
					
						
							| 
									
										
										
										
											2017-02-04 15:07:46 -08:00
										 |  |  |     if (!decoded_wstr) { | 
					
						
							| 
									
										
										
										
											2016-09-17 13:51:23 -07:00
										 |  |  |         PyErr_Clear(); | 
					
						
							|  |  |  |         return '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-04 17:36:47 -08:00
										 |  |  |     char m = '\0'; | 
					
						
							|  |  |  |     if (!_wcsicmp(decoded_wstr, L"CONIN$")) { | 
					
						
							|  |  |  |         m = 'r'; | 
					
						
							|  |  |  |     } else if (!_wcsicmp(decoded_wstr, L"CONOUT$")) { | 
					
						
							|  |  |  |         m = 'w'; | 
					
						
							|  |  |  |     } else if (!_wcsicmp(decoded_wstr, L"CON")) { | 
					
						
							|  |  |  |         m = 'x'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (m) { | 
					
						
							|  |  |  |         PyMem_Free(decoded_wstr); | 
					
						
							|  |  |  |         return m; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-04 15:07:46 -08:00
										 |  |  |     DWORD length; | 
					
						
							|  |  |  |     wchar_t name_buf[MAX_PATH], *pname_buf = name_buf; | 
					
						
							| 
									
										
										
										
											2017-06-08 18:19:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-04 15:07:46 -08:00
										 |  |  |     length = GetFullPathNameW(decoded_wstr, MAX_PATH, pname_buf, NULL); | 
					
						
							|  |  |  |     if (length > MAX_PATH) { | 
					
						
							|  |  |  |         pname_buf = PyMem_New(wchar_t, length); | 
					
						
							|  |  |  |         if (pname_buf) | 
					
						
							|  |  |  |             length = GetFullPathNameW(decoded_wstr, length, pname_buf, NULL); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             length = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     PyMem_Free(decoded_wstr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (length) { | 
					
						
							|  |  |  |         wchar_t *name = pname_buf; | 
					
						
							|  |  |  |         if (length >= 4 && name[3] == L'\\' && | 
					
						
							|  |  |  |             (name[2] == L'.' || name[2] == L'?') && | 
					
						
							|  |  |  |             name[1] == L'\\' && name[0] == L'\\') { | 
					
						
							|  |  |  |             name += 4; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!_wcsicmp(name, L"CONIN$")) { | 
					
						
							|  |  |  |             m = 'r'; | 
					
						
							|  |  |  |         } else if (!_wcsicmp(name, L"CONOUT$")) { | 
					
						
							|  |  |  |             m = 'w'; | 
					
						
							|  |  |  |         } else if (!_wcsicmp(name, L"CON")) { | 
					
						
							|  |  |  |             m = 'x'; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-04 15:07:46 -08:00
										 |  |  |     if (pname_buf != name_buf) | 
					
						
							|  |  |  |         PyMem_Free(pname_buf); | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     return m; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-04 15:07:46 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | module _io | 
					
						
							|  |  |  | class _io._WindowsConsoleIO "winconsoleio *" "&PyWindowsConsoleIO_Type" | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | /*[clinic end generated code: output=da39a3ee5e6b4b0d input=e897fdc1fba4e131]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  |     PyObject_HEAD | 
					
						
							|  |  |  |     int fd; | 
					
						
							|  |  |  |     unsigned int created : 1; | 
					
						
							|  |  |  |     unsigned int readable : 1; | 
					
						
							|  |  |  |     unsigned int writable : 1; | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     unsigned int closefd : 1; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     char finalizing; | 
					
						
							|  |  |  |     unsigned int blksize; | 
					
						
							|  |  |  |     PyObject *weakreflist; | 
					
						
							|  |  |  |     PyObject *dict; | 
					
						
							| 
									
										
										
										
											2016-10-03 09:04:58 -07:00
										 |  |  |     char buf[SMALLBUF]; | 
					
						
							|  |  |  |     wchar_t wbuf; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | } winconsoleio; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyTypeObject PyWindowsConsoleIO_Type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | _PyWindowsConsoleIO_closed(PyObject *self) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     return ((winconsoleio *)self)->fd == -1; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Returns 0 on success, -1 with exception set on failure. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | internal_close(winconsoleio *self) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     if (self->fd != -1) { | 
					
						
							|  |  |  |         if (self->closefd) { | 
					
						
							|  |  |  |             _Py_BEGIN_SUPPRESS_IPH | 
					
						
							|  |  |  |             close(self->fd); | 
					
						
							|  |  |  |             _Py_END_SUPPRESS_IPH | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |         self->fd = -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | _io._WindowsConsoleIO.close | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  | Close the console object. | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  | A closed console object cannot be used for further I/O operations. | 
					
						
							|  |  |  | close() may be called more than once without error. | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _io__WindowsConsoleIO_close_impl(winconsoleio *self) | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  | /*[clinic end generated code: output=27ef95b66c29057b input=68c4e5754f8136c2]*/ | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | { | 
					
						
							|  |  |  |     PyObject *res; | 
					
						
							|  |  |  |     PyObject *exc, *val, *tb; | 
					
						
							|  |  |  |     int rc; | 
					
						
							| 
									
										
										
										
											2022-02-08 13:39:07 -07:00
										 |  |  |     res = PyObject_CallMethodOneArg((PyObject*)&PyRawIOBase_Type, | 
					
						
							|  |  |  |                                     &_Py_ID(close), (PyObject*)self); | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     if (!self->closefd) { | 
					
						
							|  |  |  |         self->fd = -1; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         return res; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (res == NULL) | 
					
						
							|  |  |  |         PyErr_Fetch(&exc, &val, &tb); | 
					
						
							|  |  |  |     rc = internal_close(self); | 
					
						
							|  |  |  |     if (res == NULL) | 
					
						
							|  |  |  |         _PyErr_ChainExceptions(exc, val, tb); | 
					
						
							|  |  |  |     if (rc < 0) | 
					
						
							|  |  |  |         Py_CLEAR(res); | 
					
						
							|  |  |  |     return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | winconsoleio_new(PyTypeObject *type, PyObject *args, PyObject *kwds) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     winconsoleio *self; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert(type != NULL && type->tp_alloc != NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     self = (winconsoleio *) type->tp_alloc(type, 0); | 
					
						
							|  |  |  |     if (self != NULL) { | 
					
						
							|  |  |  |         self->fd = -1; | 
					
						
							|  |  |  |         self->created = 0; | 
					
						
							|  |  |  |         self->readable = 0; | 
					
						
							|  |  |  |         self->writable = 0; | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |         self->closefd = 0; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         self->blksize = 0; | 
					
						
							|  |  |  |         self->weakreflist = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return (PyObject *) self; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | _io._WindowsConsoleIO.__init__ | 
					
						
							|  |  |  |     file as nameobj: object | 
					
						
							|  |  |  |     mode: str = "r" | 
					
						
							| 
									
										
										
										
											2022-12-03 21:52:21 +02:00
										 |  |  |     closefd: bool = True | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     opener: object = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Open a console buffer by file descriptor. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | The mode can be 'rb' (default), or 'wb' for reading or writing bytes. All | 
					
						
							|  |  |  | other mode characters will be ignored. Mode 'b' will be assumed if it is | 
					
						
							|  |  |  | omitted. The *opener* parameter is always ignored. | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | _io__WindowsConsoleIO___init___impl(winconsoleio *self, PyObject *nameobj, | 
					
						
							|  |  |  |                                     const char *mode, int closefd, | 
					
						
							|  |  |  |                                     PyObject *opener) | 
					
						
							| 
									
										
										
										
											2022-12-03 21:52:21 +02:00
										 |  |  | /*[clinic end generated code: output=3fd9cbcdd8d95429 input=7a3eed6bbe998fd9]*/ | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | { | 
					
						
							|  |  |  |     const char *s; | 
					
						
							|  |  |  |     wchar_t *name = NULL; | 
					
						
							| 
									
										
										
										
											2016-09-17 13:51:23 -07:00
										 |  |  |     char console_type = '\0'; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     int ret = 0; | 
					
						
							|  |  |  |     int rwa = 0; | 
					
						
							|  |  |  |     int fd = -1; | 
					
						
							|  |  |  |     int fd_is_own = 0; | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     HANDLE handle = NULL; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-15 14:07:59 +01:00
										 |  |  |     assert(PyObject_TypeCheck(self, (PyTypeObject *)&PyWindowsConsoleIO_Type)); | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     if (self->fd >= 0) { | 
					
						
							|  |  |  |         if (self->closefd) { | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |             /* Have to close the existing file first. */ | 
					
						
							|  |  |  |             if (internal_close(self) < 0) | 
					
						
							|  |  |  |                 return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |             self->fd = -1; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fd = _PyLong_AsInt(nameobj); | 
					
						
							|  |  |  |     if (fd < 0) { | 
					
						
							|  |  |  |         if (!PyErr_Occurred()) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_ValueError, | 
					
						
							|  |  |  |                             "negative file descriptor"); | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         PyErr_Clear(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     self->fd = fd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (fd < 0) { | 
					
						
							| 
									
										
										
										
											2017-06-08 18:19:25 +02:00
										 |  |  |         PyObject *decodedname; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         int d = PyUnicode_FSDecoder(nameobj, (void*)&decodedname); | 
					
						
							|  |  |  |         if (!d) | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-27 16:03:14 +03:00
										 |  |  |         name = PyUnicode_AsWideCharString(decodedname, NULL); | 
					
						
							| 
									
										
										
										
											2016-09-17 13:51:23 -07:00
										 |  |  |         console_type = _PyIO_get_console_type(decodedname); | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         Py_CLEAR(decodedname); | 
					
						
							|  |  |  |         if (name == NULL) | 
					
						
							|  |  |  |             return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s = mode; | 
					
						
							|  |  |  |     while (*s) { | 
					
						
							|  |  |  |         switch (*s++) { | 
					
						
							|  |  |  |         case '+': | 
					
						
							|  |  |  |         case 'a': | 
					
						
							|  |  |  |         case 'b': | 
					
						
							|  |  |  |         case 'x': | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case 'r': | 
					
						
							|  |  |  |             if (rwa) | 
					
						
							|  |  |  |                 goto bad_mode; | 
					
						
							|  |  |  |             rwa = 1; | 
					
						
							|  |  |  |             self->readable = 1; | 
					
						
							| 
									
										
										
										
											2016-09-17 13:51:23 -07:00
										 |  |  |             if (console_type == 'x') | 
					
						
							|  |  |  |                 console_type = 'r'; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |             break; | 
					
						
							|  |  |  |         case 'w': | 
					
						
							|  |  |  |             if (rwa) | 
					
						
							|  |  |  |                 goto bad_mode; | 
					
						
							|  |  |  |             rwa = 1; | 
					
						
							|  |  |  |             self->writable = 1; | 
					
						
							| 
									
										
										
										
											2016-09-17 13:51:23 -07:00
										 |  |  |             if (console_type == 'x') | 
					
						
							|  |  |  |                 console_type = 'w'; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             PyErr_Format(PyExc_ValueError, | 
					
						
							|  |  |  |                          "invalid mode: %.200s", mode); | 
					
						
							|  |  |  |             goto error; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!rwa) | 
					
						
							|  |  |  |         goto bad_mode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (fd >= 0) { | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |         handle = _Py_get_osfhandle_noraise(fd); | 
					
						
							|  |  |  |         self->closefd = 0; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         DWORD access = GENERIC_READ; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |         self->closefd = 1; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         if (!closefd) { | 
					
						
							|  |  |  |             PyErr_SetString(PyExc_ValueError, | 
					
						
							|  |  |  |                 "Cannot use closefd=False with file name"); | 
					
						
							|  |  |  |             goto error; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (self->writable) | 
					
						
							| 
									
										
										
										
											2016-09-17 13:51:23 -07:00
										 |  |  |             access = GENERIC_WRITE; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         Py_BEGIN_ALLOW_THREADS | 
					
						
							|  |  |  |         /* Attempt to open for read/write initially, then fall back
 | 
					
						
							|  |  |  |            on the specific access. This is required for modern names | 
					
						
							|  |  |  |            CONIN$ and CONOUT$, which allow reading/writing state as | 
					
						
							|  |  |  |            well as reading/writing content. */ | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |         handle = CreateFileW(name, GENERIC_READ | GENERIC_WRITE, | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |             FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL); | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |         if (handle == INVALID_HANDLE_VALUE) | 
					
						
							|  |  |  |             handle = CreateFileW(name, access, | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |                 FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL); | 
					
						
							|  |  |  |         Py_END_ALLOW_THREADS | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |         if (handle == INVALID_HANDLE_VALUE) { | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |             PyErr_SetExcFromWindowsErrWithFilenameObject(PyExc_OSError, GetLastError(), nameobj); | 
					
						
							|  |  |  |             goto error; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (self->writable) | 
					
						
							|  |  |  |             self->fd = _Py_open_osfhandle_noraise(handle, _O_WRONLY | _O_BINARY); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             self->fd = _Py_open_osfhandle_noraise(handle, _O_RDONLY | _O_BINARY); | 
					
						
							|  |  |  |         if (self->fd < 0) { | 
					
						
							|  |  |  |             CloseHandle(handle); | 
					
						
							|  |  |  |             PyErr_SetFromErrnoWithFilenameObject(PyExc_OSError, nameobj); | 
					
						
							|  |  |  |             goto error; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-17 13:51:23 -07:00
										 |  |  |     if (console_type == '\0') | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |         console_type = _get_console_type(handle); | 
					
						
							| 
									
										
										
										
											2016-09-17 13:51:23 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (self->writable && console_type != 'w') { | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							|  |  |  |             "Cannot open console input buffer for writing"); | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-09-17 13:51:23 -07:00
										 |  |  |     if (self->readable && console_type != 'r') { | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         PyErr_SetString(PyExc_ValueError, | 
					
						
							|  |  |  |             "Cannot open console output buffer for reading"); | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     self->blksize = DEFAULT_BUFFER_SIZE; | 
					
						
							|  |  |  |     memset(self->buf, 0, 4); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-08 13:39:07 -07:00
										 |  |  |     if (PyObject_SetAttr((PyObject *)self, &_Py_ID(name), nameobj) < 0) | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         goto error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     goto done; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bad_mode: | 
					
						
							|  |  |  |     PyErr_SetString(PyExc_ValueError, | 
					
						
							|  |  |  |                     "Must have exactly one of read or write mode"); | 
					
						
							|  |  |  | error: | 
					
						
							|  |  |  |     ret = -1; | 
					
						
							|  |  |  |     internal_close(self); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | done: | 
					
						
							|  |  |  |     if (name) | 
					
						
							|  |  |  |         PyMem_Free(name); | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | winconsoleio_traverse(winconsoleio *self, visitproc visit, void *arg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Py_VISIT(self->dict); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | winconsoleio_clear(winconsoleio *self) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Py_CLEAR(self->dict); | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | winconsoleio_dealloc(winconsoleio *self) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     self->finalizing = 1; | 
					
						
							|  |  |  |     if (_PyIOBase_finalize((PyObject *) self) < 0) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     _PyObject_GC_UNTRACK(self); | 
					
						
							|  |  |  |     if (self->weakreflist != NULL) | 
					
						
							|  |  |  |         PyObject_ClearWeakRefs((PyObject *) self); | 
					
						
							|  |  |  |     Py_CLEAR(self->dict); | 
					
						
							|  |  |  |     Py_TYPE(self)->tp_free((PyObject *)self); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | err_closed(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     PyErr_SetString(PyExc_ValueError, "I/O operation on closed file"); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | err_mode(const char *action) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _PyIO_State *state = IO_STATE(); | 
					
						
							|  |  |  |     if (state != NULL) | 
					
						
							|  |  |  |         PyErr_Format(state->unsupported_operation, | 
					
						
							|  |  |  |                      "Console buffer does not support %s", action); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | _io._WindowsConsoleIO.fileno | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Return the underlying file descriptor (an integer). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _io__WindowsConsoleIO_fileno_impl(winconsoleio *self) | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  | /*[clinic end generated code: output=006fa74ce3b5cfbf input=845c47ebbc3a2f67]*/ | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | { | 
					
						
							|  |  |  |     if (self->fd < 0) | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |         return err_closed(); | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     return PyLong_FromLong(self->fd); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | _io._WindowsConsoleIO.readable | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | True if console is an input buffer. | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _io__WindowsConsoleIO_readable_impl(winconsoleio *self) | 
					
						
							|  |  |  | /*[clinic end generated code: output=daf9cef2743becf0 input=6be9defb5302daae]*/ | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     if (self->fd == -1) | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         return err_closed(); | 
					
						
							|  |  |  |     return PyBool_FromLong((long) self->readable); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | _io._WindowsConsoleIO.writable | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | True if console is an output buffer. | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _io__WindowsConsoleIO_writable_impl(winconsoleio *self) | 
					
						
							|  |  |  | /*[clinic end generated code: output=e0a2ad7eae5abf67 input=cefbd8abc24df6a0]*/ | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     if (self->fd == -1) | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         return err_closed(); | 
					
						
							|  |  |  |     return PyBool_FromLong((long) self->writable); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static DWORD | 
					
						
							|  |  |  | _buflen(winconsoleio *self) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-10-03 09:04:58 -07:00
										 |  |  |     for (DWORD i = 0; i < SMALLBUF; ++i) { | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         if (!self->buf[i]) | 
					
						
							|  |  |  |             return i; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-03 09:04:58 -07:00
										 |  |  |     return SMALLBUF; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static DWORD | 
					
						
							|  |  |  | _copyfrombuf(winconsoleio *self, char *buf, DWORD len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DWORD n = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (self->buf[0] && len--) { | 
					
						
							| 
									
										
										
										
											2016-10-03 09:04:58 -07:00
										 |  |  |         buf[n++] = self->buf[0]; | 
					
						
							|  |  |  |         for (int i = 1; i < SMALLBUF; ++i) | 
					
						
							|  |  |  |             self->buf[i - 1] = self->buf[i]; | 
					
						
							|  |  |  |         self->buf[SMALLBUF - 1] = 0; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static wchar_t * | 
					
						
							|  |  |  | read_console_w(HANDLE handle, DWORD maxlen, DWORD *readlen) { | 
					
						
							|  |  |  |     int err = 0, sig = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     wchar_t *buf = (wchar_t*)PyMem_Malloc(maxlen * sizeof(wchar_t)); | 
					
						
							|  |  |  |     if (!buf) | 
					
						
							|  |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2016-10-03 09:04:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     *readlen = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-03 09:04:58 -07:00
										 |  |  |     //DebugBreak();
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     Py_BEGIN_ALLOW_THREADS | 
					
						
							| 
									
										
										
										
											2016-10-03 09:04:58 -07:00
										 |  |  |     DWORD off = 0; | 
					
						
							|  |  |  |     while (off < maxlen) { | 
					
						
							| 
									
										
										
										
											2018-11-21 22:27:47 +01:00
										 |  |  |         DWORD n = (DWORD)-1; | 
					
						
							| 
									
										
										
										
											2018-07-19 15:34:03 -07:00
										 |  |  |         DWORD len = min(maxlen - off, BUFSIZ); | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         SetLastError(0); | 
					
						
							|  |  |  |         BOOL res = ReadConsoleW(handle, &buf[off], len, &n, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!res) { | 
					
						
							|  |  |  |             err = GetLastError(); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-19 15:34:03 -07:00
										 |  |  |         if (n == (DWORD)-1 && (err = GetLastError()) == ERROR_OPERATION_ABORTED) { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         if (n == 0) { | 
					
						
							|  |  |  |             err = GetLastError(); | 
					
						
							|  |  |  |             if (err != ERROR_OPERATION_ABORTED) | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             err = 0; | 
					
						
							|  |  |  |             HANDLE hInterruptEvent = _PyOS_SigintEvent(); | 
					
						
							|  |  |  |             if (WaitForSingleObjectEx(hInterruptEvent, 100, FALSE) | 
					
						
							| 
									
										
										
										
											2016-10-03 09:04:58 -07:00
										 |  |  |                     == WAIT_OBJECT_0) { | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |                 ResetEvent(hInterruptEvent); | 
					
						
							|  |  |  |                 Py_BLOCK_THREADS | 
					
						
							|  |  |  |                 sig = PyErr_CheckSignals(); | 
					
						
							|  |  |  |                 Py_UNBLOCK_THREADS | 
					
						
							|  |  |  |                 if (sig < 0) | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         *readlen += n; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* If we didn't read a full buffer that time, don't try
 | 
					
						
							|  |  |  |            again or we will block a second time. */ | 
					
						
							|  |  |  |         if (n < len) | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         /* If the buffer ended with a newline, break out */ | 
					
						
							|  |  |  |         if (buf[*readlen - 1] == '\n') | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2016-10-03 09:04:58 -07:00
										 |  |  |         /* If the buffer ends with a high surrogate, expand the
 | 
					
						
							|  |  |  |            buffer and read an extra character. */ | 
					
						
							|  |  |  |         WORD char_type; | 
					
						
							|  |  |  |         if (off + BUFSIZ >= maxlen && | 
					
						
							|  |  |  |             GetStringTypeW(CT_CTYPE3, &buf[*readlen - 1], 1, &char_type) && | 
					
						
							|  |  |  |             char_type == C3_HIGHSURROGATE) { | 
					
						
							|  |  |  |             wchar_t *newbuf; | 
					
						
							|  |  |  |             maxlen += 1; | 
					
						
							|  |  |  |             Py_BLOCK_THREADS | 
					
						
							|  |  |  |             newbuf = (wchar_t*)PyMem_Realloc(buf, maxlen * sizeof(wchar_t)); | 
					
						
							|  |  |  |             Py_UNBLOCK_THREADS | 
					
						
							|  |  |  |             if (!newbuf) { | 
					
						
							|  |  |  |                 sig = -1; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             buf = newbuf; | 
					
						
							|  |  |  |             /* Only advance by n and not BUFSIZ in this case */ | 
					
						
							|  |  |  |             off += n; | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         off += BUFSIZ; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-10-03 09:04:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     Py_END_ALLOW_THREADS | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (sig) | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     if (err) { | 
					
						
							|  |  |  |         PyErr_SetFromWindowsErr(err); | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (*readlen > 0 && buf[0] == L'\x1a') { | 
					
						
							|  |  |  |         PyMem_Free(buf); | 
					
						
							|  |  |  |         buf = (wchar_t *)PyMem_Malloc(sizeof(wchar_t)); | 
					
						
							|  |  |  |         if (!buf) | 
					
						
							|  |  |  |             goto error; | 
					
						
							|  |  |  |         buf[0] = L'\0'; | 
					
						
							|  |  |  |         *readlen = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return buf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | error: | 
					
						
							|  |  |  |     if (buf) | 
					
						
							|  |  |  |         PyMem_Free(buf); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Py_ssize_t | 
					
						
							|  |  |  | readinto(winconsoleio *self, char *buf, Py_ssize_t len) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     if (self->fd == -1) { | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         err_closed(); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!self->readable) { | 
					
						
							|  |  |  |         err_mode("reading"); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (len == 0) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     if (len > BUFMAX) { | 
					
						
							|  |  |  |         PyErr_Format(PyExc_ValueError, "cannot read more than %d bytes", BUFMAX); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     HANDLE handle = _Py_get_osfhandle(self->fd); | 
					
						
							|  |  |  |     if (handle == INVALID_HANDLE_VALUE) | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     /* Each character may take up to 4 bytes in the final buffer.
 | 
					
						
							|  |  |  |        This is highly conservative, but necessary to avoid | 
					
						
							|  |  |  |        failure for any given Unicode input (e.g. \U0010ffff). | 
					
						
							|  |  |  |        If the caller requests fewer than 4 bytes, we buffer one | 
					
						
							|  |  |  |        character. | 
					
						
							|  |  |  |     */ | 
					
						
							|  |  |  |     DWORD wlen = (DWORD)(len / 4); | 
					
						
							|  |  |  |     if (wlen == 0) { | 
					
						
							|  |  |  |         wlen = 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DWORD read_len = _copyfrombuf(self, buf, (DWORD)len); | 
					
						
							|  |  |  |     if (read_len) { | 
					
						
							|  |  |  |         buf = &buf[read_len]; | 
					
						
							|  |  |  |         len -= read_len; | 
					
						
							|  |  |  |         wlen -= 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (len == read_len || wlen == 0) | 
					
						
							|  |  |  |         return read_len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DWORD n; | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     wchar_t *wbuf = read_console_w(handle, wlen, &n); | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     if (wbuf == NULL) | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     if (n == 0) { | 
					
						
							|  |  |  |         PyMem_Free(wbuf); | 
					
						
							|  |  |  |         return read_len; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int err = 0; | 
					
						
							|  |  |  |     DWORD u8n = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Py_BEGIN_ALLOW_THREADS | 
					
						
							|  |  |  |     if (len < 4) { | 
					
						
							|  |  |  |         if (WideCharToMultiByte(CP_UTF8, 0, wbuf, n, | 
					
						
							|  |  |  |                 self->buf, sizeof(self->buf) / sizeof(self->buf[0]), | 
					
						
							|  |  |  |                 NULL, NULL)) | 
					
						
							|  |  |  |             u8n = _copyfrombuf(self, buf, (DWORD)len); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         u8n = WideCharToMultiByte(CP_UTF8, 0, wbuf, n, | 
					
						
							|  |  |  |             buf, (DWORD)len, NULL, NULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (u8n) { | 
					
						
							|  |  |  |         read_len += u8n; | 
					
						
							|  |  |  |         u8n = 0; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         err = GetLastError(); | 
					
						
							|  |  |  |         if (err == ERROR_INSUFFICIENT_BUFFER) { | 
					
						
							|  |  |  |             /* Calculate the needed buffer for a more useful error, as this
 | 
					
						
							|  |  |  |                 means our "/ 4" logic above is insufficient for some input. | 
					
						
							|  |  |  |             */ | 
					
						
							|  |  |  |             u8n = WideCharToMultiByte(CP_UTF8, 0, wbuf, n, | 
					
						
							|  |  |  |                 NULL, 0, NULL, NULL); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_END_ALLOW_THREADS | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PyMem_Free(wbuf); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (u8n) { | 
					
						
							|  |  |  |         PyErr_Format(PyExc_SystemError, | 
					
						
							| 
									
										
										
										
											2019-03-13 22:59:55 +02:00
										 |  |  |             "Buffer had room for %zd bytes but %u bytes required", | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |             len, u8n); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (err) { | 
					
						
							|  |  |  |         PyErr_SetFromWindowsErr(err); | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return read_len; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | _io._WindowsConsoleIO.readinto | 
					
						
							|  |  |  |     buffer: Py_buffer(accept={rwbuffer}) | 
					
						
							|  |  |  |     / | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Same as RawIOBase.readinto(). | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _io__WindowsConsoleIO_readinto_impl(winconsoleio *self, Py_buffer *buffer) | 
					
						
							|  |  |  | /*[clinic end generated code: output=66d1bdfa3f20af39 input=4ed68da48a6baffe]*/ | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Py_ssize_t len = readinto(self, buffer->buf, buffer->len); | 
					
						
							|  |  |  |     if (len < 0) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return PyLong_FromSsize_t(len); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static DWORD | 
					
						
							|  |  |  | new_buffersize(winconsoleio *self, DWORD currentsize) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DWORD addend; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Expand the buffer by an amount proportional to the current size,
 | 
					
						
							|  |  |  |        giving us amortized linear-time behavior.  For bigger sizes, use a | 
					
						
							|  |  |  |        less-than-double growth factor to avoid excessive allocation. */ | 
					
						
							|  |  |  |     if (currentsize > 65536) | 
					
						
							|  |  |  |         addend = currentsize >> 3; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         addend = 256 + currentsize; | 
					
						
							|  |  |  |     if (addend < SMALLCHUNK) | 
					
						
							|  |  |  |         /* Avoid tiny read() calls. */ | 
					
						
							|  |  |  |         addend = SMALLCHUNK; | 
					
						
							|  |  |  |     return addend + currentsize; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | _io._WindowsConsoleIO.readall | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Read all data from the console, returned as bytes. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Return an empty bytes object at EOF. | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _io__WindowsConsoleIO_readall_impl(winconsoleio *self) | 
					
						
							|  |  |  | /*[clinic end generated code: output=e6d312c684f6e23b input=4024d649a1006e69]*/ | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     wchar_t *buf; | 
					
						
							|  |  |  |     DWORD bufsize, n, len = 0; | 
					
						
							|  |  |  |     PyObject *bytes; | 
					
						
							|  |  |  |     DWORD bytes_size, rn; | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     HANDLE handle; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     if (self->fd == -1) | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         return err_closed(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     handle = _Py_get_osfhandle(self->fd); | 
					
						
							|  |  |  |     if (handle == INVALID_HANDLE_VALUE) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     bufsize = BUFSIZ; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     buf = (wchar_t*)PyMem_Malloc((bufsize + 1) * sizeof(wchar_t)); | 
					
						
							|  |  |  |     if (buf == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (1) { | 
					
						
							|  |  |  |         wchar_t *subbuf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (len >= (Py_ssize_t)bufsize) { | 
					
						
							|  |  |  |             DWORD newsize = new_buffersize(self, len); | 
					
						
							|  |  |  |             if (newsize > BUFMAX) | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             if (newsize < bufsize) { | 
					
						
							|  |  |  |                 PyErr_SetString(PyExc_OverflowError, | 
					
						
							|  |  |  |                                 "unbounded read returned more bytes " | 
					
						
							|  |  |  |                                 "than a Python bytes object can hold"); | 
					
						
							|  |  |  |                 PyMem_Free(buf); | 
					
						
							|  |  |  |                 return NULL; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             bufsize = newsize; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-07 03:11:30 -07:00
										 |  |  |             wchar_t *tmp = PyMem_Realloc(buf, | 
					
						
							|  |  |  |                                          (bufsize + 1) * sizeof(wchar_t)); | 
					
						
							|  |  |  |             if (tmp == NULL) { | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |                 PyMem_Free(buf); | 
					
						
							|  |  |  |                 return NULL; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-12-07 03:11:30 -07:00
										 |  |  |             buf = tmp; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |         subbuf = read_console_w(handle, bufsize - len, &n); | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (subbuf == NULL) { | 
					
						
							|  |  |  |             PyMem_Free(buf); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (n > 0) | 
					
						
							|  |  |  |             wcsncpy_s(&buf[len], bufsize - len + 1, subbuf, n); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         PyMem_Free(subbuf); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-08 12:37:33 -07:00
										 |  |  |         /* when the read is empty we break */ | 
					
						
							|  |  |  |         if (n == 0) | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         len += n; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-08 12:37:33 -07:00
										 |  |  |     if (len == 0 && _buflen(self) == 0) { | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         /* when the result starts with ^Z we return an empty buffer */ | 
					
						
							|  |  |  |         PyMem_Free(buf); | 
					
						
							|  |  |  |         return PyBytes_FromStringAndSize(NULL, 0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-08 12:37:33 -07:00
										 |  |  |     if (len) { | 
					
						
							|  |  |  |         Py_BEGIN_ALLOW_THREADS | 
					
						
							|  |  |  |         bytes_size = WideCharToMultiByte(CP_UTF8, 0, buf, len, | 
					
						
							|  |  |  |             NULL, 0, NULL, NULL); | 
					
						
							|  |  |  |         Py_END_ALLOW_THREADS | 
					
						
							| 
									
										
										
										
											2016-09-19 22:17:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-08 12:37:33 -07:00
										 |  |  |         if (!bytes_size) { | 
					
						
							|  |  |  |             DWORD err = GetLastError(); | 
					
						
							|  |  |  |             PyMem_Free(buf); | 
					
						
							|  |  |  |             return PyErr_SetFromWindowsErr(err); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         bytes_size = 0; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bytes_size += _buflen(self); | 
					
						
							|  |  |  |     bytes = PyBytes_FromStringAndSize(NULL, bytes_size); | 
					
						
							|  |  |  |     rn = _copyfrombuf(self, PyBytes_AS_STRING(bytes), bytes_size); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-08 12:37:33 -07:00
										 |  |  |     if (len) { | 
					
						
							|  |  |  |         Py_BEGIN_ALLOW_THREADS | 
					
						
							|  |  |  |         bytes_size = WideCharToMultiByte(CP_UTF8, 0, buf, len, | 
					
						
							|  |  |  |             &PyBytes_AS_STRING(bytes)[rn], bytes_size - rn, NULL, NULL); | 
					
						
							|  |  |  |         Py_END_ALLOW_THREADS | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-08 12:37:33 -07:00
										 |  |  |         if (!bytes_size) { | 
					
						
							|  |  |  |             DWORD err = GetLastError(); | 
					
						
							|  |  |  |             PyMem_Free(buf); | 
					
						
							|  |  |  |             Py_CLEAR(bytes); | 
					
						
							|  |  |  |             return PyErr_SetFromWindowsErr(err); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* add back the number of preserved bytes */ | 
					
						
							|  |  |  |         bytes_size += rn; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PyMem_Free(buf); | 
					
						
							|  |  |  |     if (bytes_size < (size_t)PyBytes_GET_SIZE(bytes)) { | 
					
						
							|  |  |  |         if (_PyBytes_Resize(&bytes, n * sizeof(wchar_t)) < 0) { | 
					
						
							|  |  |  |             Py_CLEAR(bytes); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return bytes; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | _io._WindowsConsoleIO.read | 
					
						
							| 
									
										
										
										
											2017-03-30 09:15:31 +03:00
										 |  |  |     size: Py_ssize_t(accept={int, NoneType}) = -1 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     / | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Read at most size bytes, returned as bytes. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Only makes one system call when size is a positive integer, | 
					
						
							|  |  |  | so less data may be returned than requested. | 
					
						
							|  |  |  | Return an empty bytes object at EOF. | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _io__WindowsConsoleIO_read_impl(winconsoleio *self, Py_ssize_t size) | 
					
						
							| 
									
										
										
										
											2017-03-30 09:15:31 +03:00
										 |  |  | /*[clinic end generated code: output=57df68af9f4b22d0 input=8bc73bc15d0fa072]*/ | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | { | 
					
						
							|  |  |  |     PyObject *bytes; | 
					
						
							|  |  |  |     Py_ssize_t bytes_size; | 
					
						
							| 
									
										
										
										
											2016-09-19 22:17:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     if (self->fd == -1) | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         return err_closed(); | 
					
						
							|  |  |  |     if (!self->readable) | 
					
						
							|  |  |  |         return err_mode("reading"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (size < 0) | 
					
						
							|  |  |  |         return _io__WindowsConsoleIO_readall_impl(self); | 
					
						
							|  |  |  |     if (size > BUFMAX) { | 
					
						
							|  |  |  |         PyErr_Format(PyExc_ValueError, "cannot read more than %d bytes", BUFMAX); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bytes = PyBytes_FromStringAndSize(NULL, size); | 
					
						
							|  |  |  |     if (bytes == NULL) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bytes_size = readinto(self, PyBytes_AS_STRING(bytes), PyBytes_GET_SIZE(bytes)); | 
					
						
							|  |  |  |     if (bytes_size < 0) { | 
					
						
							|  |  |  |         Py_CLEAR(bytes); | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (bytes_size < PyBytes_GET_SIZE(bytes)) { | 
					
						
							|  |  |  |         if (_PyBytes_Resize(&bytes, bytes_size) < 0) { | 
					
						
							|  |  |  |             Py_CLEAR(bytes); | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return bytes; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | _io._WindowsConsoleIO.write | 
					
						
							|  |  |  |     b: Py_buffer | 
					
						
							|  |  |  |     / | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Write buffer b to file, return number of bytes written. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Only makes one system call, so not all of the data may be written. | 
					
						
							|  |  |  | The number of bytes actually written is returned. | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _io__WindowsConsoleIO_write_impl(winconsoleio *self, Py_buffer *b) | 
					
						
							|  |  |  | /*[clinic end generated code: output=775bdb16fbf9137b input=be35fb624f97c941]*/ | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     BOOL res = TRUE; | 
					
						
							|  |  |  |     wchar_t *wbuf; | 
					
						
							| 
									
										
										
										
											2023-01-17 19:23:06 +00:00
										 |  |  |     DWORD len, wlen, orig_len, n = 0; | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     HANDLE handle; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     if (self->fd == -1) | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         return err_closed(); | 
					
						
							|  |  |  |     if (!self->writable) | 
					
						
							|  |  |  |         return err_mode("writing"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     handle = _Py_get_osfhandle(self->fd); | 
					
						
							|  |  |  |     if (handle == INVALID_HANDLE_VALUE) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-24 18:55:51 +02:00
										 |  |  |     if (!b->len) { | 
					
						
							|  |  |  |         return PyLong_FromLong(0); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     if (b->len > BUFMAX) | 
					
						
							|  |  |  |         len = BUFMAX; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         len = (DWORD)b->len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Py_BEGIN_ALLOW_THREADS | 
					
						
							|  |  |  |     wlen = MultiByteToWideChar(CP_UTF8, 0, b->buf, len, NULL, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* issue11395 there is an unspecified upper bound on how many bytes
 | 
					
						
							|  |  |  |        can be written at once. We cap at 32k - the caller will have to | 
					
						
							|  |  |  |        handle partial writes. | 
					
						
							|  |  |  |        Since we don't know how many input bytes are being ignored, we | 
					
						
							|  |  |  |        have to reduce and recalculate. */ | 
					
						
							|  |  |  |     while (wlen > 32766 / sizeof(wchar_t)) { | 
					
						
							|  |  |  |         len /= 2; | 
					
						
							| 
									
										
										
										
											2023-01-17 19:23:06 +00:00
										 |  |  |         orig_len = len; | 
					
						
							|  |  |  |         /* Reduce the length until we hit the final byte of a UTF-8 sequence
 | 
					
						
							|  |  |  |          * (top bit is unset). Fix for github issue 82052. | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         while (len > 0 && (((char *)b->buf)[len-1] & 0x80) != 0) | 
					
						
							|  |  |  |             --len; | 
					
						
							|  |  |  |         /* If we hit a length of 0, something has gone wrong. This shouldn't
 | 
					
						
							|  |  |  |          * be possible, as valid UTF-8 can have at most 3 non-final bytes | 
					
						
							|  |  |  |          * before a final one, and our buffer is way longer than that. | 
					
						
							|  |  |  |          * But to be on the safe side, if we hit this issue we just restore | 
					
						
							|  |  |  |          * the original length and let the console API sort it out. | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         if (len == 0) { | 
					
						
							|  |  |  |             len = orig_len; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         wlen = MultiByteToWideChar(CP_UTF8, 0, b->buf, len, NULL, 0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Py_END_ALLOW_THREADS | 
					
						
							| 
									
										
										
										
											2016-09-19 22:17:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     if (!wlen) | 
					
						
							|  |  |  |         return PyErr_SetFromWindowsErr(0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     wbuf = (wchar_t*)PyMem_Malloc(wlen * sizeof(wchar_t)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Py_BEGIN_ALLOW_THREADS | 
					
						
							|  |  |  |     wlen = MultiByteToWideChar(CP_UTF8, 0, b->buf, len, wbuf, wlen); | 
					
						
							|  |  |  |     if (wlen) { | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |         res = WriteConsoleW(handle, wbuf, wlen, &n, NULL); | 
					
						
							| 
									
										
										
										
											2017-06-02 19:26:01 +03:00
										 |  |  |         if (res && n < wlen) { | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |             /* Wrote fewer characters than expected, which means our
 | 
					
						
							|  |  |  |              * len value may be wrong. So recalculate it from the | 
					
						
							|  |  |  |              * characters that were written. As this could potentially | 
					
						
							|  |  |  |              * result in a different value, we also validate that value. | 
					
						
							|  |  |  |              */ | 
					
						
							|  |  |  |             len = WideCharToMultiByte(CP_UTF8, 0, wbuf, n, | 
					
						
							|  |  |  |                 NULL, 0, NULL, NULL); | 
					
						
							|  |  |  |             if (len) { | 
					
						
							|  |  |  |                 wlen = MultiByteToWideChar(CP_UTF8, 0, b->buf, len, | 
					
						
							|  |  |  |                     NULL, 0); | 
					
						
							|  |  |  |                 assert(wlen == len); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else | 
					
						
							|  |  |  |         res = 0; | 
					
						
							|  |  |  |     Py_END_ALLOW_THREADS | 
					
						
							| 
									
										
										
										
											2016-09-19 22:17:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     if (!res) { | 
					
						
							|  |  |  |         DWORD err = GetLastError(); | 
					
						
							|  |  |  |         PyMem_Free(wbuf); | 
					
						
							|  |  |  |         return PyErr_SetFromWindowsErr(err); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PyMem_Free(wbuf); | 
					
						
							|  |  |  |     return PyLong_FromSsize_t(len); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | winconsoleio_repr(winconsoleio *self) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     if (self->fd == -1) | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         return PyUnicode_FromFormat("<_io._WindowsConsoleIO [closed]>"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (self->readable) | 
					
						
							|  |  |  |         return PyUnicode_FromFormat("<_io._WindowsConsoleIO mode='rb' closefd=%s>", | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |             self->closefd ? "True" : "False"); | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     if (self->writable) | 
					
						
							|  |  |  |         return PyUnicode_FromFormat("<_io._WindowsConsoleIO mode='wb' closefd=%s>", | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |             self->closefd ? "True" : "False"); | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     PyErr_SetString(PyExc_SystemError, "_WindowsConsoleIO has invalid mode"); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*[clinic input]
 | 
					
						
							|  |  |  | _io._WindowsConsoleIO.isatty | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Always True. | 
					
						
							|  |  |  | [clinic start generated code]*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | _io__WindowsConsoleIO_isatty_impl(winconsoleio *self) | 
					
						
							|  |  |  | /*[clinic end generated code: output=9eac09d287c11bd7 input=9b91591dbe356f86]*/ | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     if (self->fd == -1) | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |         return err_closed(); | 
					
						
							| 
									
										
										
										
											2016-09-19 22:17:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     Py_RETURN_TRUE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "clinic/winconsoleio.c.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyMethodDef winconsoleio_methods[] = { | 
					
						
							|  |  |  |     _IO__WINDOWSCONSOLEIO_READ_METHODDEF | 
					
						
							|  |  |  |     _IO__WINDOWSCONSOLEIO_READALL_METHODDEF | 
					
						
							|  |  |  |     _IO__WINDOWSCONSOLEIO_READINTO_METHODDEF | 
					
						
							|  |  |  |     _IO__WINDOWSCONSOLEIO_WRITE_METHODDEF | 
					
						
							|  |  |  |     _IO__WINDOWSCONSOLEIO_CLOSE_METHODDEF | 
					
						
							|  |  |  |     _IO__WINDOWSCONSOLEIO_READABLE_METHODDEF | 
					
						
							|  |  |  |     _IO__WINDOWSCONSOLEIO_WRITABLE_METHODDEF | 
					
						
							|  |  |  |     _IO__WINDOWSCONSOLEIO_FILENO_METHODDEF | 
					
						
							|  |  |  |     _IO__WINDOWSCONSOLEIO_ISATTY_METHODDEF | 
					
						
							|  |  |  |     {NULL,           NULL}             /* sentinel */ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* 'closed' and 'mode' are attributes for compatibility with FileIO. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | get_closed(winconsoleio *self, void *closure) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     return PyBool_FromLong((long)(self->fd == -1)); | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | get_closefd(winconsoleio *self, void *closure) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-04-24 01:00:27 +03:00
										 |  |  |     return PyBool_FromLong((long)(self->closefd)); | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | get_mode(winconsoleio *self, void *closure) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return PyUnicode_FromString(self->readable ? "rb" : "wb"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyGetSetDef winconsoleio_getsetlist[] = { | 
					
						
							|  |  |  |     {"closed", (getter)get_closed, NULL, "True if the file is closed"}, | 
					
						
							|  |  |  |     {"closefd", (getter)get_closefd, NULL, | 
					
						
							|  |  |  |         "True if the file descriptor will be closed by close()."}, | 
					
						
							|  |  |  |     {"mode", (getter)get_mode, NULL, "String giving the file mode"}, | 
					
						
							|  |  |  |     {NULL}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static PyMemberDef winconsoleio_members[] = { | 
					
						
							|  |  |  |     {"_blksize", T_UINT, offsetof(winconsoleio, blksize), 0}, | 
					
						
							|  |  |  |     {"_finalizing", T_BOOL, offsetof(winconsoleio, finalizing), 0}, | 
					
						
							|  |  |  |     {NULL} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | PyTypeObject PyWindowsConsoleIO_Type = { | 
					
						
							|  |  |  |     PyVarObject_HEAD_INIT(NULL, 0) | 
					
						
							|  |  |  |     "_io._WindowsConsoleIO", | 
					
						
							|  |  |  |     sizeof(winconsoleio), | 
					
						
							|  |  |  |     0, | 
					
						
							|  |  |  |     (destructor)winconsoleio_dealloc,           /* tp_dealloc */ | 
					
						
							| 
									
										
										
										
											2019-05-31 04:13:39 +02:00
										 |  |  |     0,                                          /* tp_vectorcall_offset */ | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     0,                                          /* tp_getattr */ | 
					
						
							|  |  |  |     0,                                          /* tp_setattr */ | 
					
						
							| 
									
										
										
										
											2019-05-31 04:13:39 +02:00
										 |  |  |     0,                                          /* tp_as_async */ | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     (reprfunc)winconsoleio_repr,                /* tp_repr */ | 
					
						
							|  |  |  |     0,                                          /* tp_as_number */ | 
					
						
							|  |  |  |     0,                                          /* tp_as_sequence */ | 
					
						
							|  |  |  |     0,                                          /* tp_as_mapping */ | 
					
						
							|  |  |  |     0,                                          /* tp_hash */ | 
					
						
							|  |  |  |     0,                                          /* tp_call */ | 
					
						
							|  |  |  |     0,                                          /* tp_str */ | 
					
						
							|  |  |  |     PyObject_GenericGetAttr,                    /* tp_getattro */ | 
					
						
							|  |  |  |     0,                                          /* tp_setattro */ | 
					
						
							|  |  |  |     0,                                          /* tp_as_buffer */ | 
					
						
							|  |  |  |     Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE | 
					
						
							| 
									
										
										
										
											2019-05-29 22:12:38 +02:00
										 |  |  |         | Py_TPFLAGS_HAVE_GC,                   /* tp_flags */ | 
					
						
							| 
									
										
										
										
											2016-08-30 21:22:36 -07:00
										 |  |  |     _io__WindowsConsoleIO___init____doc__,      /* tp_doc */ | 
					
						
							|  |  |  |     (traverseproc)winconsoleio_traverse,        /* tp_traverse */ | 
					
						
							|  |  |  |     (inquiry)winconsoleio_clear,                /* tp_clear */ | 
					
						
							|  |  |  |     0,                                          /* tp_richcompare */ | 
					
						
							|  |  |  |     offsetof(winconsoleio, weakreflist),        /* tp_weaklistoffset */ | 
					
						
							|  |  |  |     0,                                          /* tp_iter */ | 
					
						
							|  |  |  |     0,                                          /* tp_iternext */ | 
					
						
							|  |  |  |     winconsoleio_methods,                       /* tp_methods */ | 
					
						
							|  |  |  |     winconsoleio_members,                       /* tp_members */ | 
					
						
							|  |  |  |     winconsoleio_getsetlist,                    /* tp_getset */ | 
					
						
							|  |  |  |     0,                                          /* tp_base */ | 
					
						
							|  |  |  |     0,                                          /* tp_dict */ | 
					
						
							|  |  |  |     0,                                          /* tp_descr_get */ | 
					
						
							|  |  |  |     0,                                          /* tp_descr_set */ | 
					
						
							|  |  |  |     offsetof(winconsoleio, dict),               /* tp_dictoffset */ | 
					
						
							|  |  |  |     _io__WindowsConsoleIO___init__,             /* tp_init */ | 
					
						
							|  |  |  |     PyType_GenericAlloc,                        /* tp_alloc */ | 
					
						
							|  |  |  |     winconsoleio_new,                           /* tp_new */ | 
					
						
							|  |  |  |     PyObject_GC_Del,                            /* tp_free */ | 
					
						
							|  |  |  |     0,                                          /* tp_is_gc */ | 
					
						
							|  |  |  |     0,                                          /* tp_bases */ | 
					
						
							|  |  |  |     0,                                          /* tp_mro */ | 
					
						
							|  |  |  |     0,                                          /* tp_cache */ | 
					
						
							|  |  |  |     0,                                          /* tp_subclasses */ | 
					
						
							|  |  |  |     0,                                          /* tp_weaklist */ | 
					
						
							|  |  |  |     0,                                          /* tp_del */ | 
					
						
							|  |  |  |     0,                                          /* tp_version_tag */ | 
					
						
							|  |  |  |     0,                                          /* tp_finalize */ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* MS_WINDOWS */
 |