barrier/src/lib/server/Server.h

478 lines
14 KiB
C
Raw Normal View History

2012-06-10 16:50:54 +00:00
/*
* synergy -- mouse and keyboard sharing utility
* Copyright (C) 2012 Synergy Si Ltd.
* Copyright (C) 2002 Chris Schoeneman
2012-06-10 16:50:54 +00:00
*
* This package is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* found in the file COPYING that should have accompanied this file.
*
* This package is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#pragma once
#include "server/Config.h"
#include "synergy/clipboard_types.h"
#include "synergy/Clipboard.h"
#include "synergy/key_types.h"
#include "synergy/mouse_types.h"
#include "synergy/INode.h"
#include "synergy/DragInformation.h"
#include "base/Event.h"
#include "base/Stopwatch.h"
#include "base/EventTypes.h"
#include "common/stdmap.h"
#include "common/stdset.h"
#include "common/stdvector.h"
2012-06-10 16:50:54 +00:00
2014-11-11 13:51:47 +00:00
class BaseClientProxy;
class EventQueueTimer;
class PrimaryClient;
class InputFilter;
namespace synergy { class Screen; }
class IEventQueue;
2014-11-11 13:51:47 +00:00
class Thread;
2012-06-10 16:50:54 +00:00
//! Synergy server
/*!
This class implements the top-level server algorithms for synergy.
*/
2014-11-11 13:51:47 +00:00
class Server : public INode {
2012-06-10 16:50:54 +00:00
public:
//! Lock cursor to screen data
2014-11-11 13:51:47 +00:00
class LockCursorToScreenInfo {
2012-06-10 16:50:54 +00:00
public:
enum State { kOff, kOn, kToggle };
2014-11-11 13:51:47 +00:00
static LockCursorToScreenInfo* alloc(State state = kToggle);
2012-06-10 16:50:54 +00:00
public:
State m_state;
};
//! Switch to screen data
2014-11-11 13:51:47 +00:00
class SwitchToScreenInfo {
2012-06-10 16:50:54 +00:00
public:
2014-11-11 13:51:47 +00:00
static SwitchToScreenInfo* alloc(const String& screen);
2012-06-10 16:50:54 +00:00
public:
// this is a C-string; this type is a variable size structure
char m_screen[1];
};
//! Switch in direction data
2014-11-11 13:51:47 +00:00
class SwitchInDirectionInfo {
2012-06-10 16:50:54 +00:00
public:
2014-11-11 13:51:47 +00:00
static SwitchInDirectionInfo* alloc(EDirection direction);
2012-06-10 16:50:54 +00:00
public:
EDirection m_direction;
};
//! Screen connected data
2014-11-11 13:51:47 +00:00
class ScreenConnectedInfo {
2012-06-10 16:50:54 +00:00
public:
2014-11-11 13:51:47 +00:00
ScreenConnectedInfo(String screen) : m_screen(screen) { }
2012-06-10 16:50:54 +00:00
public:
2014-11-11 13:51:47 +00:00
String m_screen; // was char[1]
2012-06-10 16:50:54 +00:00
};
//! Keyboard broadcast data
2014-11-11 13:51:47 +00:00
class KeyboardBroadcastInfo {
2012-06-10 16:50:54 +00:00
public:
enum State { kOff, kOn, kToggle };
2014-11-11 13:51:47 +00:00
static KeyboardBroadcastInfo* alloc(State state = kToggle);
static KeyboardBroadcastInfo* alloc(State state,
const String& screens);
2012-06-10 16:50:54 +00:00
public:
State m_state;
char m_screens[1];
};
/*!
Start the server with the configuration \p config and the primary
client (local screen) \p primaryClient. The client retains
ownership of \p primaryClient.
*/
Server(Config& config, PrimaryClient* primaryClient,
synergy::Screen* screen, IEventQueue* events, bool enableDragDrop);
2014-11-11 13:51:47 +00:00
~Server();
#ifdef TEST_ENV
2014-11-11 13:51:47 +00:00
Server() : m_mock(true), m_config(NULL) { }
void setActive(BaseClientProxy* active) { m_active = active; }
#endif
2012-06-10 16:50:54 +00:00
//! @name manipulators
//@{
//! Set configuration
/*!
Change the server's configuration. Returns true iff the new
configuration was accepted (it must include the server's name).
This will disconnect any clients no longer in the configuration.
*/
2014-11-11 13:51:47 +00:00
bool setConfig(const Config&);
2012-06-10 16:50:54 +00:00
//! Add a client
/*!
Adds \p client to the server. The client is adopted and will be
destroyed when the client disconnects or is disconnected.
*/
2014-11-11 13:51:47 +00:00
void adoptClient(BaseClientProxy* client);
2012-06-10 16:50:54 +00:00
//! Disconnect clients
/*!
Disconnect clients. This tells them to disconnect but does not wait
for them to actually do so. The server sends the disconnected event
when they're all disconnected (or immediately if none are connected).
The caller can also just destroy this object to force the disconnection.
*/
void disconnect();
//! Clears the file buffer
void clearReceivedFileData();
//! Set the expected size of receiving file
2014-11-11 13:51:47 +00:00
void setExpectedFileSize(String data);
//! Received a chunk of file data
2014-11-11 13:51:47 +00:00
void fileChunkReceived(String data);
//! Create a new thread and use it to send file to client
void sendFileToClient(const char* filename);
//! Received dragging information from client
2014-11-11 13:51:47 +00:00
void dragInfoReceived(UInt32 fileNum, String content);
2012-06-10 16:50:54 +00:00
//@}
//! @name accessors
//@{
//! Get number of connected clients
/*!
Returns the number of connected clients, including the server itself.
*/
UInt32 getNumClients() const;
//! Get the list of connected clients
/*!
Set the \c list to the names of the currently connected clients.
*/
2014-11-11 13:51:47 +00:00
void getClients(std::vector<String>& list) const;
//! Return true if recieved file size is valid
bool isReceivedFileSizeValid();
2012-06-10 16:50:54 +00:00
//! Return expected file size
size_t getExpectedFileSize() { return m_expectedFileSize; }
2012-06-10 16:50:54 +00:00
//@}
private:
// get canonical name of client
2014-11-11 13:51:47 +00:00
String getName(const BaseClientProxy*) const;
2012-06-10 16:50:54 +00:00
// get the sides of the primary screen that have neighbors
UInt32 getActivePrimarySides() const;
// returns true iff mouse should be locked to the current screen
// according to this object only, ignoring what the primary client
// says.
bool isLockedToScreenServer() const;
// returns true iff mouse should be locked to the current screen
// according to this object or the primary client.
bool isLockedToScreen() const;
// returns the jump zone of the client
2014-11-11 13:51:47 +00:00
SInt32 getJumpZoneSize(BaseClientProxy*) const;
2012-06-10 16:50:54 +00:00
// change the active screen
2014-11-11 13:51:47 +00:00
void switchScreen(BaseClientProxy*,
2012-06-10 16:50:54 +00:00
SInt32 x, SInt32 y, bool forScreenSaver);
// jump to screen
2014-11-11 13:51:47 +00:00
void jumpToScreen(BaseClientProxy*);
2012-06-10 16:50:54 +00:00
// convert pixel position to fraction, using x or y depending on the
// direction.
2014-11-11 13:51:47 +00:00
float mapToFraction(BaseClientProxy*, EDirection,
2012-06-10 16:50:54 +00:00
SInt32 x, SInt32 y) const;
// convert fraction to pixel position, writing only x or y depending
// on the direction.
2014-11-11 13:51:47 +00:00
void mapToPixel(BaseClientProxy*, EDirection, float f,
2012-06-10 16:50:54 +00:00
SInt32& x, SInt32& y) const;
// returns true if the client has a neighbor anywhere along the edge
// indicated by the direction.
2014-11-11 13:51:47 +00:00
bool hasAnyNeighbor(BaseClientProxy*, EDirection) const;
2012-06-10 16:50:54 +00:00
// lookup neighboring screen, mapping the coordinate independent of
// the direction to the neighbor's coordinate space.
2014-11-11 13:51:47 +00:00
BaseClientProxy* getNeighbor(BaseClientProxy*, EDirection,
2012-06-10 16:50:54 +00:00
SInt32& x, SInt32& y) const;
// lookup neighboring screen. given a position relative to the
// source screen, find the screen we should move onto and where.
// if the position is sufficiently far from the source then we
// cross multiple screens. if there is no suitable screen then
// return NULL and x,y are not modified.
2014-11-11 13:51:47 +00:00
BaseClientProxy* mapToNeighbor(BaseClientProxy*, EDirection,
2012-06-10 16:50:54 +00:00
SInt32& x, SInt32& y) const;
// adjusts x and y or neither to avoid ending up in a jump zone
// after entering the client in the given direction.
2014-11-11 13:51:47 +00:00
void avoidJumpZone(BaseClientProxy*, EDirection,
2012-06-10 16:50:54 +00:00
SInt32& x, SInt32& y) const;
// test if a switch is permitted. this includes testing user
// options like switch delay and tracking any state required to
// implement them. returns true iff a switch is permitted.
2014-11-11 13:51:47 +00:00
bool isSwitchOkay(BaseClientProxy* dst, EDirection,
2012-06-10 16:50:54 +00:00
SInt32 x, SInt32 y, SInt32 xActive, SInt32 yActive);
// update switch state due to a mouse move at \p x, \p y that
// doesn't switch screens.
void noSwitch(SInt32 x, SInt32 y);
// stop switch timers
void stopSwitch();
// start two tap switch timer
void startSwitchTwoTap();
// arm the two tap switch timer if \p x, \p y is outside the tap zone
void armSwitchTwoTap(SInt32 x, SInt32 y);
// stop the two tap switch timer
void stopSwitchTwoTap();
// returns true iff the two tap switch timer is started
bool isSwitchTwoTapStarted() const;
// returns true iff should switch because of two tap
bool shouldSwitchTwoTap() const;
// start delay switch timer
void startSwitchWait(SInt32 x, SInt32 y);
// stop delay switch timer
void stopSwitchWait();
// returns true iff the delay switch timer is started
bool isSwitchWaitStarted() const;
// returns the corner (EScreenSwitchCornerMasks) where x,y is on the
// given client. corners have the given size.
2014-11-11 13:51:47 +00:00
UInt32 getCorner(BaseClientProxy*,
2012-06-10 16:50:54 +00:00
SInt32 x, SInt32 y, SInt32 size) const;
// stop relative mouse moves
void stopRelativeMoves();
// send screen options to \c client
2014-11-11 13:51:47 +00:00
void sendOptions(BaseClientProxy* client) const;
2012-06-10 16:50:54 +00:00
// process options from configuration
void processOptions();
// event handlers
2014-11-11 13:51:47 +00:00
void handleShapeChanged(const Event&, void*);
void handleClipboardGrabbed(const Event&, void*);
void handleClipboardChanged(const Event&, void*);
void handleKeyDownEvent(const Event&, void*);
void handleKeyUpEvent(const Event&, void*);
void handleKeyRepeatEvent(const Event&, void*);
void handleButtonDownEvent(const Event&, void*);
void handleButtonUpEvent(const Event&, void*);
void handleMotionPrimaryEvent(const Event&, void*);
void handleMotionSecondaryEvent(const Event&, void*);
void handleWheelEvent(const Event&, void*);
void handleScreensaverActivatedEvent(const Event&, void*);
void handleScreensaverDeactivatedEvent(const Event&, void*);
void handleSwitchWaitTimeout(const Event&, void*);
void handleClientDisconnected(const Event&, void*);
void handleClientCloseTimeout(const Event&, void*);
void handleSwitchToScreenEvent(const Event&, void*);
void handleSwitchInDirectionEvent(const Event&, void*);
void handleKeyboardBroadcastEvent(const Event&,void*);
void handleLockCursorToScreenEvent(const Event&, void*);
void handleFakeInputBeginEvent(const Event&, void*);
void handleFakeInputEndEvent(const Event&, void*);
void handleFileChunkSendingEvent(const Event&, void*);
void handleFileRecieveCompletedEvent(const Event&, void*);
2012-06-10 16:50:54 +00:00
// event processing
2014-11-11 13:51:47 +00:00
void onClipboardChanged(BaseClientProxy* sender,
2012-06-10 16:50:54 +00:00
ClipboardID id, UInt32 seqNum);
void onScreensaver(bool activated);
void onKeyDown(KeyID, KeyModifierMask, KeyButton,
const char* screens);
void onKeyUp(KeyID, KeyModifierMask, KeyButton,
const char* screens);
void onKeyRepeat(KeyID, KeyModifierMask, SInt32, KeyButton);
void onMouseDown(ButtonID);
void onMouseUp(ButtonID);
bool onMouseMovePrimary(SInt32 x, SInt32 y);
void onMouseMoveSecondary(SInt32 dx, SInt32 dy);
void onMouseWheel(SInt32 xDelta, SInt32 yDelta);
void onFileChunkSending(const void* data);
void onFileRecieveCompleted();
2012-06-10 16:50:54 +00:00
// add client to list and attach event handlers for client
2014-11-11 13:51:47 +00:00
bool addClient(BaseClientProxy*);
2012-06-10 16:50:54 +00:00
// remove client from list and detach event handlers for client
2014-11-11 13:51:47 +00:00
bool removeClient(BaseClientProxy*);
2012-06-10 16:50:54 +00:00
// close a client
2014-11-11 13:51:47 +00:00
void closeClient(BaseClientProxy*, const char* msg);
2012-06-10 16:50:54 +00:00
// close clients not in \p config
2014-11-11 13:51:47 +00:00
void closeClients(const Config& config);
2012-06-10 16:50:54 +00:00
// close all clients whether they've completed the handshake or not,
// except the primary client
void closeAllClients();
// remove clients from internal state
2014-11-11 13:51:47 +00:00
void removeActiveClient(BaseClientProxy*);
void removeOldClient(BaseClientProxy*);
2012-06-10 16:50:54 +00:00
// force the cursor off of \p client
2014-11-11 13:51:47 +00:00
void forceLeaveClient(BaseClientProxy* client);
// thread funciton for sending file
void sendFileThread(void*);
// thread function for writing file to drop directory
void writeToDropDirThread(void*);
// thread function for getting drag filename
void getDragInfoThread(void*);
// send drag info to new client screen
2014-11-11 13:51:47 +00:00
void sendDragInfo(BaseClientProxy* newScreen);
public:
bool m_mock;
2012-06-10 16:50:54 +00:00
private:
2014-11-11 13:51:47 +00:00
class ClipboardInfo {
2012-06-10 16:50:54 +00:00
public:
2014-11-11 13:51:47 +00:00
ClipboardInfo();
2012-06-10 16:50:54 +00:00
public:
2014-11-11 13:51:47 +00:00
Clipboard m_clipboard;
String m_clipboardData;
String m_clipboardOwner;
2012-06-10 16:50:54 +00:00
UInt32 m_clipboardSeqNum;
};
// the primary screen client
2014-11-11 13:51:47 +00:00
PrimaryClient* m_primaryClient;
2012-06-10 16:50:54 +00:00
// all clients (including the primary client) indexed by name
2014-11-11 13:51:47 +00:00
typedef std::map<String, BaseClientProxy*> ClientList;
typedef std::set<BaseClientProxy*> ClientSet;
ClientList m_clients;
ClientSet m_clientSet;
2012-06-10 16:50:54 +00:00
// all old connections that we're waiting to hangup
2014-11-11 13:51:47 +00:00
typedef std::map<BaseClientProxy*, EventQueueTimer*> OldClients;
OldClients m_oldClients;
2012-06-10 16:50:54 +00:00
// the client with focus
2014-11-11 13:51:47 +00:00
BaseClientProxy* m_active;
2012-06-10 16:50:54 +00:00
// the sequence number of enter messages
UInt32 m_seqNum;
// current mouse position (in absolute screen coordinates) on
// whichever screen is active
SInt32 m_x, m_y;
// last mouse deltas. this is needed to smooth out double tap
// on win32 which reports bogus mouse motion at the edge of
// the screen when using low level hooks, synthesizing motion
// in the opposite direction the mouse actually moved.
SInt32 m_xDelta, m_yDelta;
SInt32 m_xDelta2, m_yDelta2;
// current configuration
2014-11-11 13:51:47 +00:00
Config* m_config;
2012-06-10 16:50:54 +00:00
// input filter (from m_config);
2014-11-11 13:51:47 +00:00
InputFilter* m_inputFilter;
2012-06-10 16:50:54 +00:00
// clipboard cache
2014-11-11 13:51:47 +00:00
ClipboardInfo m_clipboards[kClipboardEnd];
2012-06-10 16:50:54 +00:00
// state saved when screen saver activates
2014-11-11 13:51:47 +00:00
BaseClientProxy* m_activeSaver;
2012-06-10 16:50:54 +00:00
SInt32 m_xSaver, m_ySaver;
// common state for screen switch tests. all tests are always
// trying to reach the same screen in the same direction.
EDirection m_switchDir;
2014-11-11 13:51:47 +00:00
BaseClientProxy* m_switchScreen;
2012-06-10 16:50:54 +00:00
// state for delayed screen switching
double m_switchWaitDelay;
2014-11-11 13:51:47 +00:00
EventQueueTimer* m_switchWaitTimer;
2012-06-10 16:50:54 +00:00
SInt32 m_switchWaitX, m_switchWaitY;
// state for double-tap screen switching
double m_switchTwoTapDelay;
2014-11-11 13:51:47 +00:00
Stopwatch m_switchTwoTapTimer;
2012-06-10 16:50:54 +00:00
bool m_switchTwoTapEngaged;
bool m_switchTwoTapArmed;
SInt32 m_switchTwoTapZone;
// modifiers needed before switching
bool m_switchNeedsShift;
bool m_switchNeedsControl;
bool m_switchNeedsAlt;
// relative mouse move option
bool m_relativeMoves;
// flag whether or not we have broadcasting enabled and the screens to
// which we should send broadcasted keys.
bool m_keyboardBroadcasting;
2014-11-11 13:51:47 +00:00
String m_keyboardBroadcastingScreens;
2012-06-10 16:50:54 +00:00
// screen locking (former scroll lock)
bool m_lockedToScreen;
// server screen
synergy::Screen* m_screen;
2012-06-10 16:50:54 +00:00
IEventQueue* m_events;
// file transfer
size_t m_expectedFileSize;
2014-11-11 13:51:47 +00:00
String m_receivedFileData;
DragFileList m_dragFileList;
Thread* m_sendFileThread;
Thread* m_writeToDropDirThread;
String m_dragFileExt;
bool m_ignoreFileTransfer;
bool m_enableDragDrop;
2014-11-11 13:51:47 +00:00
Thread* m_getDragInfoThread;
bool m_waitDragInfoThread;
2012-06-10 16:50:54 +00:00
};