162 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			162 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
/* fhandler_windows.cc: code to access windows message queues.
 | 
						|
 | 
						|
   Written by Sergey S. Okhapkin (sos@prospect.com.ru).
 | 
						|
   Feedback and testing by Andy Piper (andyp@parallax.co.uk).
 | 
						|
 | 
						|
This file is part of Cygwin.
 | 
						|
 | 
						|
This software is a copyrighted work licensed under the terms of the
 | 
						|
Cygwin license.  Please consult the file "CYGWIN_LICENSE" for
 | 
						|
details. */
 | 
						|
 | 
						|
#include "winsup.h"
 | 
						|
#include "cygerrno.h"
 | 
						|
#include "path.h"
 | 
						|
#include "fhandler.h"
 | 
						|
#include "sigproc.h"
 | 
						|
#include "thread.h"
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
The following unix-style calls are supported:
 | 
						|
 | 
						|
	open ("/dev/windows", flags, mode=0)
 | 
						|
		- create a unix fd for message queue.
 | 
						|
 | 
						|
	read (fd, buf, len)
 | 
						|
		- return next message from queue. buf must point to MSG
 | 
						|
		  structure, len must be >= sizeof (MSG). If read is set to
 | 
						|
		  non-blocking and the queue is empty, read call returns -1
 | 
						|
		  immediately with errno set to EAGAIN, otherwise it blocks
 | 
						|
		  untill the message will be received.
 | 
						|
 | 
						|
	write (fd, buf, len)
 | 
						|
		- send a message pointed by buf. len argument ignored.
 | 
						|
 | 
						|
	ioctl (fd, command, *param)
 | 
						|
		- control read()/write() behavior.
 | 
						|
		ioctl (fd, WINDOWS_POST, NULL): write() will PostMessage();
 | 
						|
		ioctl (fd, WINDOWS_SEND, NULL): write() will SendMessage();
 | 
						|
		ioctl (fd, WINDOWS_HWND, &hWnd): read() messages for
 | 
						|
			hWnd window.
 | 
						|
 | 
						|
	select () call marks read fd when any message posted to queue.
 | 
						|
*/
 | 
						|
 | 
						|
fhandler_windows::fhandler_windows ()
 | 
						|
  : fhandler_base (), hWnd_ (NULL), method_ (WINDOWS_POST)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
fhandler_windows::open (int flags, mode_t mode)
 | 
						|
{
 | 
						|
  return fhandler_base::open ((flags & ~O_TEXT) | O_BINARY, mode);
 | 
						|
}
 | 
						|
 | 
						|
ssize_t
 | 
						|
fhandler_windows::write (const void *buf, size_t)
 | 
						|
{
 | 
						|
  MSG *ptr = (MSG *) buf;
 | 
						|
 | 
						|
  if (method_ == WINDOWS_POST)
 | 
						|
    {
 | 
						|
      if (!PostMessageW (ptr->hwnd, ptr->message, ptr->wParam, ptr->lParam))
 | 
						|
	{
 | 
						|
	  __seterrno ();
 | 
						|
	  return -1;
 | 
						|
	}
 | 
						|
    }
 | 
						|
  else if (!SendNotifyMessageW (ptr->hwnd, ptr->message, ptr->wParam,
 | 
						|
				ptr->lParam))
 | 
						|
    {
 | 
						|
      __seterrno ();
 | 
						|
      return -1;
 | 
						|
    }
 | 
						|
  return sizeof (MSG);
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
fhandler_windows::read (void *buf, size_t& len)
 | 
						|
{
 | 
						|
  MSG *ptr = (MSG *) buf;
 | 
						|
 | 
						|
  if (len < sizeof (MSG))
 | 
						|
    {
 | 
						|
      set_errno (EINVAL);
 | 
						|
      len = (size_t) -1;
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
  HANDLE w4[2];
 | 
						|
  wait_signal_arrived here (w4[0]);
 | 
						|
  DWORD cnt = 1;
 | 
						|
  if ((w4[1] = pthread::get_cancel_event ()) != NULL)
 | 
						|
    ++cnt;
 | 
						|
  for (;;)
 | 
						|
    {
 | 
						|
      switch (MsgWaitForMultipleObjectsEx (cnt, w4,
 | 
						|
					   is_nonblocking () ? 0 : INFINITE,
 | 
						|
					   QS_ALLINPUT | QS_ALLPOSTMESSAGE,
 | 
						|
					   MWMO_INPUTAVAILABLE))
 | 
						|
	{
 | 
						|
	case WAIT_OBJECT_0:
 | 
						|
	  if (_my_tls.call_signal_handler ())
 | 
						|
	    continue;
 | 
						|
	  len = (size_t) -1;
 | 
						|
	  set_errno (EINTR);
 | 
						|
	  break;
 | 
						|
	case WAIT_OBJECT_0 + 1:
 | 
						|
	  if (cnt > 1) /* WAIT_OBJECT_0 + 1 is the cancel event object. */
 | 
						|
	    {
 | 
						|
	      pthread::static_cancel_self ();
 | 
						|
	      break;
 | 
						|
	    }
 | 
						|
	  fallthrough;
 | 
						|
	case WAIT_OBJECT_0 + 2:
 | 
						|
	  if (!PeekMessageW (ptr, hWnd_, 0, 0, PM_REMOVE))
 | 
						|
	    {
 | 
						|
	      len = (size_t) -1;
 | 
						|
	      __seterrno ();
 | 
						|
	    }
 | 
						|
	  else if (ptr->message == WM_QUIT)
 | 
						|
	    len = 0;
 | 
						|
	  else
 | 
						|
	    len = sizeof (MSG);
 | 
						|
	  break;
 | 
						|
	case WAIT_TIMEOUT:
 | 
						|
	  len = (size_t) -1;
 | 
						|
	  set_errno (EAGAIN);
 | 
						|
	  break;
 | 
						|
	default:
 | 
						|
	  len = (size_t) -1;
 | 
						|
	  __seterrno ();
 | 
						|
	  break;
 | 
						|
	}
 | 
						|
      break;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
int
 | 
						|
fhandler_windows::ioctl (unsigned int cmd, void *val)
 | 
						|
{
 | 
						|
  switch (cmd)
 | 
						|
    {
 | 
						|
    case WINDOWS_POST:
 | 
						|
    case WINDOWS_SEND:
 | 
						|
      method_ = cmd;
 | 
						|
      break;
 | 
						|
    case WINDOWS_HWND:
 | 
						|
      if (val == NULL)
 | 
						|
	{
 | 
						|
	  set_errno (EINVAL);
 | 
						|
	  return -1;
 | 
						|
	}
 | 
						|
      hWnd_ = * ((HWND *) val);
 | 
						|
      break;
 | 
						|
    default:
 | 
						|
      return fhandler_base::ioctl (cmd, val);
 | 
						|
    }
 | 
						|
  return 0;
 | 
						|
}
 |