|
|
#include "precomp.h"
//
// CONTROL.CPP
// Control by us, control of us
//
// Copyright(c) Microsoft 1997-
//
#define MLZ_FILE_ZONE ZONE_CORE
//
// CA_ReceivedPacket()
//
void ASShare::CA_ReceivedPacket ( ASPerson * pasFrom, PS20DATAPACKET pPacket ) { PCAPACKET pCAPacket;
DebugEntry(ASShare::CA_ReceivedPacket);
ValidatePerson(pasFrom);
pCAPacket = (PCAPACKET)pPacket;
switch (pCAPacket->msg) { case CA_MSG_NOTIFY_STATE: if (pasFrom->cpcCaps.general.version < CAPS_VERSION_30) { ERROR_OUT(("Ignoring CA_MSG_NOTIFY_STATE from 2.x node [%d]", pasFrom->mcsID)); } else { CAHandleNewState(pasFrom, (PCANOTPACKET)pPacket); } break;
case CA_OLDMSG_DETACH: case CA_OLDMSG_COOPERATE: // Set "cooperating", and map it to allow/disallow control
CA2xCooperateChange(pasFrom, (pCAPacket->msg == CA_OLDMSG_COOPERATE)); break;
case CA_OLDMSG_REQUEST_CONTROL: CA2xRequestControl(pasFrom, pCAPacket); break;
case CA_OLDMSG_GRANTED_CONTROL: CA2xGrantedControl(pasFrom, pCAPacket); break;
default: // Ignore for now -- old 2.x messages
break; }
DebugExitVOID(ASShare::CA_ReceivedPacket); }
//
// CA30_ReceivedPacket()
//
void ASShare::CA30_ReceivedPacket ( ASPerson * pasFrom, PS20DATAPACKET pPacket ) { LPBYTE pCAPacket;
DebugEntry(ASShare::CA30_ReceivedPacket);
pCAPacket = (LPBYTE)pPacket + sizeof(CA30PACKETHEADER);
if (pasFrom->cpcCaps.general.version < CAPS_VERSION_30) { ERROR_OUT(("Ignoring CA30 packet %d from 2.x node [%d]", ((PCA30PACKETHEADER)pPacket)->msg, pasFrom->mcsID)); DC_QUIT; }
switch (((PCA30PACKETHEADER)pPacket)->msg) { // From VIEWER (remote) to HOST (us)
case CA_REQUEST_TAKECONTROL: { CAHandleRequestTakeControl(pasFrom, (PCA_RTC_PACKET)pCAPacket); break; }
// From HOST (remote) to VIEWER (us)
case CA_REPLY_REQUEST_TAKECONTROL: { CAHandleReplyRequestTakeControl(pasFrom, (PCA_REPLY_RTC_PACKET)pCAPacket); break; }
// From HOST (remote) to VIEWER (us)
case CA_REQUEST_GIVECONTROL: { CAHandleRequestGiveControl(pasFrom, (PCA_RGC_PACKET)pCAPacket); break; }
// From VIEWER (remote) to HOST (us)
case CA_REPLY_REQUEST_GIVECONTROL: { CAHandleReplyRequestGiveControl(pasFrom, (PCA_REPLY_RGC_PACKET)pCAPacket); break; }
// From CONTROLLER (remote) to HOST (us)
case CA_PREFER_PASSCONTROL: { CAHandlePreferPassControl(pasFrom, (PCA_PPC_PACKET)pCAPacket); break; }
// From CONTROLLER (remote) to HOST (us)
case CA_INFORM_RELEASEDCONTROL: { CAHandleInformReleasedControl(pasFrom, (PCA_INFORM_PACKET)pCAPacket); break; }
// From HOST (remote) to CONTROLLER (us)
case CA_INFORM_REVOKEDCONTROL: { CAHandleInformRevokedControl(pasFrom, (PCA_INFORM_PACKET)pCAPacket); break; }
// From HOST (remote) to CONTROLLER (us)
case CA_INFORM_PAUSEDCONTROL: { CAHandleInformPausedControl(pasFrom, (PCA_INFORM_PACKET)pCAPacket); break; }
// From HOST (remote) to CONTROLLER (us)
case CA_INFORM_UNPAUSEDCONTROL: { CAHandleInformUnpausedControl(pasFrom, (PCA_INFORM_PACKET)pCAPacket); break; }
default: { WARNING_OUT(("CA30_ReceivedPacket: unrecognized message %d", ((PCA30PACKETHEADER)pPacket)->msg)); break; } }
DC_EXIT_POINT: DebugExitVOID(ASShare::CA30_ReceivedPacket); }
//
// CANewRequestID()
//
// Returns a new token. It uses the current value, fills in the new one, and
// also returns the new one. We wrap around if necessary. ZERO is never
// valid. Note that this is a unique identifier only to us.
//
// It is a stamp for the control operation. Since you can't be controlling
// and controlled at the same time, we have one stamp for all ops.
//
UINT ASShare::CANewRequestID(void) { DebugEntry(ASShare::CANewRequestID);
++(m_pasLocal->m_caControlID); if (m_pasLocal->m_caControlID == 0) { ++(m_pasLocal->m_caControlID); }
DebugExitDWORD(ASShare::CANewRequestID, m_pasLocal->m_caControlID); return(m_pasLocal->m_caControlID); }
//
// CA_ViewStarting()
// Called when a REMOTE starts hosting
//
// We only do anything if it's a 2.x node since they could be cooperating
// but not hosting.
//
BOOL ASShare::CA_ViewStarting(ASPerson * pasPerson) { DebugEntry(ASShare::CA_ViewStarting);
//
// If this isn't a back level system, ignore it.
//
if (pasPerson->cpcCaps.general.version >= CAPS_VERSION_30) { DC_QUIT; }
//
// See if AllowControl should now be on.
//
if (pasPerson->m_ca2xCooperating) { //
// Yes, it should. 2.x node is cooperating, now they are hosting,
// and we can take control of them.
//
ASSERT(!pasPerson->m_caAllowControl); pasPerson->m_caAllowControl = TRUE; VIEW_HostStateChange(pasPerson); }
DC_EXIT_POINT: DebugExitBOOL(ASShare::CA_ViewStarting, TRUE); return(TRUE); }
//
// CA_ViewEnded()
// Called when a REMOTE stopped hosting
//
void ASShare::CA_ViewEnded(ASPerson * pasPerson) { PCAREQUEST pRequest; PCAREQUEST pNext;
DebugEntry(ASShare::CA_ViewEnded);
//
// Clear any control stuff we are a part of where they are the host
//
CA_ClearLocalState(CACLEAR_VIEW, pasPerson, FALSE);
//
// Clear any control stuff involving remotes
//
if (pasPerson->m_caControlledBy) { ASSERT(pasPerson->m_caControlledBy != m_pasLocal);
CAClearHostState(pasPerson, NULL); ASSERT(!pasPerson->m_caControlledBy); }
pasPerson->m_caAllowControl = FALSE;
//
// Clean up outstanding control packets to this person
//
pRequest = (PCAREQUEST)COM_BasedListFirst(&m_caQueuedMsgs, FIELD_OFFSET(CAREQUEST, chain)); while (pRequest) { pNext = (PCAREQUEST)COM_BasedListNext(&m_caQueuedMsgs, pRequest, FIELD_OFFSET(CAREQUEST, chain));
if (pRequest->destID == pasPerson->mcsID) { if (pRequest->type == REQUEST_30) { //
// Delete messages sent by us to this person who is hosting
//
switch (pRequest->msg) { case CA_REQUEST_TAKECONTROL: case CA_PREFER_PASSCONTROL: case CA_REPLY_REQUEST_GIVECONTROL: WARNING_OUT(("Deleting viewer control message %d, person [%d] stopped hosting", pRequest->msg, pasPerson->mcsID)); COM_BasedListRemove(&pRequest->chain); delete pRequest; break; } } else { ASSERT(pRequest->type == REQUEST_2X);
// Change GRANTED_CONTROL packets to this host to DETACH
if (pRequest->msg == CA_OLDMSG_GRANTED_CONTROL) { //
// For 2.x messages, destID is only non-zero when we are
// attempting to control a particular node. It allows us
// to undo/cancel control, to map our one-to-one model
// into the global 2.x collaboration model.
//
//
// Make this a DETACH, that way we don't have to worry if
// part of the COOPERATE/GRANTED_CONTROL sequence got out
// but part was left in the queue.
//
WARNING_OUT(("Changing GRANTED_CONTROL to 2.x host [%d] into DETATCH", pasPerson->mcsID));
pRequest->destID = 0; pRequest->msg = CA_OLDMSG_DETACH; pRequest->req.req2x.data1 = 0; pRequest->req.req2x.data2 = 0; } } }
pRequest = pNext; }
DebugExitVOID(ASView::CA_ViewEnded); }
//
// CA_PartyLeftShare()
//
void ASShare::CA_PartyLeftShare(ASPerson * pasPerson) { DebugEntry(ASShare::CA_PartyLeftShare);
ValidatePerson(pasPerson);
//
// Clean up 2.x control stuff
//
if (pasPerson == m_ca2xControlTokenOwner) { m_ca2xControlTokenOwner = NULL; }
//
// We must have cleaned up hosting info for this person already.
// So it can't be controlled or controllable.
//
ASSERT(!pasPerson->m_caAllowControl); ASSERT(!pasPerson->m_caControlledBy);
if (pasPerson != m_pasLocal) { PCAREQUEST pRequest; PCAREQUEST pNext;
//
// Clear any control stuff we are a part of where they are the
// viewer.
//
CA_ClearLocalState(CACLEAR_HOST, pasPerson, FALSE);
//
// Clear any control stuff involving remotes
//
if (pasPerson->m_caInControlOf) { ASSERT(pasPerson->m_caInControlOf != m_pasLocal); CAClearHostState(pasPerson->m_caInControlOf, NULL); }
//
// Clean up outgoing packets meant for this person.
//
pRequest = (PCAREQUEST)COM_BasedListFirst(&m_caQueuedMsgs, FIELD_OFFSET(CAREQUEST, chain)); while (pRequest) { pNext = (PCAREQUEST)COM_BasedListNext(&m_caQueuedMsgs, pRequest, FIELD_OFFSET(CAREQUEST, chain));
//
// This doesn't need to know if it's a 2.x or 3.0 request,
// simply remove queued packets intended for somebody leaving.
//
// Only GRANTED_CONTROL requests will have non-zero destIDs of
// the 2.x packets.
//
if (pRequest->destID == pasPerson->mcsID) { WARNING_OUT(("Freeing outgoing RESPONSE to node [%d]", pasPerson->mcsID));
COM_BasedListRemove(&(pRequest->chain)); delete pRequest; }
pRequest = pNext; }
ASSERT(m_caWaitingForReplyFrom != pasPerson); } else { //
// When our waiting for/controlled dude stopped sharing, we should
// have cleaned this goop up.
//
ASSERT(!pasPerson->m_caInControlOf); ASSERT(!pasPerson->m_caControlledBy); ASSERT(!m_caWaitingForReplyFrom); ASSERT(!m_caWaitingForReplyMsg);
//
// There should be NO outgoing control requests
//
ASSERT(COM_BasedListIsEmpty(&(m_caQueuedMsgs))); }
DebugExitVOID(ASShare::CA_PartyLeftShare); }
//
// CA_Periodic() -> SHARE STUFF
//
void ASShare::CA_Periodic(void) { DebugEntry(ASShare::CA_Periodic);
//
// Flush as many queued outgoing messages as we can
//
CAFlushOutgoingPackets();
DebugExitVOID(ASShare::CA_Periodic); }
//
// CA_SyncAlreadyHosting()
//
void ASHost::CA_SyncAlreadyHosting(void) { DebugEntry(ASHost::CA_SyncAlreadyHosting);
m_caRetrySendState = TRUE;
DebugExitVOID(ASHost::CA_SyncAlreadyHosting); }
//
// CA_Periodic() -> HOSTING STUFF
//
void ASHost::CA_Periodic(void) { DebugEntry(ASHost::CA_Periodic);
if (m_caRetrySendState) { PCANOTPACKET pPacket; #ifdef _DEBUG
UINT sentSize; #endif // _DEBUG
pPacket = (PCANOTPACKET)m_pShare->SC_AllocPkt(PROT_STR_MISC, g_s20BroadcastID, sizeof(*pPacket)); if (!pPacket) { WARNING_OUT(("CA_Periodic: couldn't broadcast new state")); } else { pPacket->header.data.dataType = DT_CA; pPacket->msg = CA_MSG_NOTIFY_STATE;
pPacket->state = 0; if (m_pShare->m_pasLocal->m_caAllowControl) pPacket->state |= CASTATE_ALLOWCONTROL;
if (m_pShare->m_pasLocal->m_caControlledBy) pPacket->controllerID = m_pShare->m_pasLocal->m_caControlledBy->mcsID; else pPacket->controllerID = 0;
#ifdef _DEBUG
sentSize = #endif // _DEBUG
m_pShare->DCS_CompressAndSendPacket(PROT_STR_MISC, g_s20BroadcastID, &(pPacket->header), sizeof(*pPacket));
m_caRetrySendState = FALSE; } }
DebugExitVOID(ASHost::CA_Periodic); }
//
// CAFlushOutgoingPackets()
//
// This tries to send private packets (not broadcast notifications) that
// we have accumulated. It returns TRUE if the outgoing queue is empty.
//
BOOL ASShare::CAFlushOutgoingPackets(void) { BOOL fEmpty = TRUE; PCAREQUEST pRequest;
//
// If we're hosting and haven't yet flushed the HET or CA state,
// force queueing.
//
if (m_hetRetrySendState || (m_pHost && m_pHost->m_caRetrySendState)) { TRACE_OUT(("CAFlushOutgoingPackets: force queuing, pending HET/CA state broadcast")); fEmpty = FALSE; DC_QUIT; }
while (pRequest = (PCAREQUEST)COM_BasedListFirst(&m_caQueuedMsgs, FIELD_OFFSET(CAREQUEST, chain))) { //
// Allocate/send packet
//
if (pRequest->type == REQUEST_30) { if (!CASendPacket(pRequest->destID, pRequest->msg, &pRequest->req.req30.packet)) { WARNING_OUT(("CAFlushOutgoingPackets: couldn't send request")); fEmpty = FALSE; break; } } else { ASSERT(pRequest->type == REQUEST_2X);
if (!CA2xSendMsg(pRequest->destID, pRequest->msg, pRequest->req.req2x.data1, pRequest->req.req2x.data2)) { WARNING_OUT(("CAFlushOutgoingmsgs: couldn't send request")); fEmpty = FALSE; break; } }
//
// Do we do state transitions here or when things are added to queue?
// requestID, results are calculated when put on queue. Results can
// change though based on a future action.
//
COM_BasedListRemove(&(pRequest->chain)); delete pRequest; }
DC_EXIT_POINT: DebugExitBOOL(CAFlushOutgoingPackets, fEmpty); return(fEmpty); }
//
// CASendPacket()
// This sends a private message (request or response) to the destination.
// If there are queued private messages in front of this one, or we can't
// send it, we add it to the pending queue.
//
// This TRUE if sent.
//
// It's up to the caller to change state info appropriately.
//
BOOL ASShare::CASendPacket ( UINT_PTR destID, UINT msg, PCA30P pData ) { BOOL fSent = FALSE; PCA30PACKETHEADER pPacket; #ifdef _DEBUG
UINT sentSize; #endif // _DEBUG
DebugEntry(ASShare::CASendPacket);
//
// Note that CA30P does not include size of header.
//
pPacket = (PCA30PACKETHEADER)SC_AllocPkt(PROT_STR_INPUT, destID, sizeof(CA30PACKETHEADER) + sizeof(*pData)); if (!pPacket) { WARNING_OUT(("CASendPacket: no memory to send %d packet to [%d]", msg, destID)); DC_QUIT; }
pPacket->header.data.dataType = DT_CA30; pPacket->msg = msg; memcpy(pPacket+1, pData, sizeof(*pData));
#ifdef _DEBUG
sentSize = #endif // _DEBUG
DCS_CompressAndSendPacket(PROT_STR_INPUT, destID, &(pPacket->header), sizeof(*pPacket)); TRACE_OUT(("CA30 request packet size: %08d, sent %08d", sizeof(*pPacket), sentSize));
fSent = TRUE;
DC_EXIT_POINT:
DebugExitBOOL(ASShare::CASendPacket, fSent); return(fSent); }
//
// CAQueueSendPacket()
// This flushes pending queued requests if there are any, then tries to
// send this one. If it can't, we add it to the queue. If there's not any
// memory even for that, we return an error about it.
//
BOOL ASShare::CAQueueSendPacket ( UINT_PTR destID, UINT msg, PCA30P pPacketSend ) { BOOL rc = TRUE; PCAREQUEST pCARequest;
DebugEntry(ASShare::CAQueueSendPacket);
//
// These must go out in order. So if any queued messages are still
// present, those must be sent first.
//
if (!CAFlushOutgoingPackets() || !CASendPacket(destID, msg, pPacketSend)) { //
// We must queue this.
//
TRACE_OUT(("CAQueueSendPacket: queuing request for send later"));
pCARequest = new CAREQUEST; if (!pCARequest) { ERROR_OUT(("CAQueueSendPacket: can't even allocate memory to queue request; must fail")); rc = FALSE; } else { SET_STAMP(pCARequest, CAREQUEST);
pCARequest->type = REQUEST_30; pCARequest->destID = destID; pCARequest->msg = msg; pCARequest->req.req30.packet = *pPacketSend;
//
// Stick this at the end of the queue
//
COM_BasedListInsertBefore(&(m_caQueuedMsgs), &(pCARequest->chain)); } }
DebugExitBOOL(ASShare::CAQueueSendPacket, rc); return(rc); }
//
// CALangToggle()
//
// This temporarily turns off the keyboard language toggle key, so that a
// remote controlling us doesn't inadvertently change it. When we stop being
// controlled, we put it back.
//
void ASShare::CALangToggle(BOOL fBackOn) { //
// Local Variables
//
LONG rc; HKEY hkeyToggle; BYTE regValue[2]; DWORD cbRegValue; DWORD dwType; LPCSTR szValue;
DebugEntry(ASShare::CALangToggle);
szValue = (g_asWin95) ? NULL : LANGUAGE_TOGGLE_KEY_VAL;
if (fBackOn) { //
// We are gaining control of our local keyboard again - we restore the
// language togging functionality.
//
// We must directly access the registry to accomplish this.
//
if (m_caToggle != LANGUAGE_TOGGLE_NOT_PRESENT) { rc = RegOpenKey(HKEY_CURRENT_USER, LANGUAGE_TOGGLE_KEY, &hkeyToggle);
if (rc == ERROR_SUCCESS) { //
// Clear the value for this key.
//
regValue[0] = m_caToggle; regValue[1] = '\0'; // ensure NUL termination
//
// Restore the value.
//
RegSetValueEx(hkeyToggle, szValue, 0, REG_SZ, regValue, sizeof(regValue));
//
// We need to inform the system about this change. We do not
// tell any other apps about this (ie do not set any of the
// notification flags as the last parm)
//
SystemParametersInfo(SPI_SETLANGTOGGLE, 0, 0, 0); }
RegCloseKey(hkeyToggle); } } else { //
// We are losing control of our keyboard - ensure that remote key
// events will not change our local keyboard settings by disabling the
// keyboard language toggle.
//
// We must directly access the registry to accomplish this.
//
rc = RegOpenKey(HKEY_CURRENT_USER, LANGUAGE_TOGGLE_KEY, &hkeyToggle);
if (rc == ERROR_SUCCESS) { cbRegValue = sizeof(regValue);
rc = RegQueryValueEx(hkeyToggle, szValue, NULL, &dwType, regValue, &cbRegValue);
if (rc == ERROR_SUCCESS) { m_caToggle = regValue[0];
//
// Clear the value for this key.
//
regValue[0] = '3'; regValue[1] = '\0'; // ensure NUL termination
//
// Clear the value.
//
RegSetValueEx(hkeyToggle, szValue, 0, REG_SZ, regValue, sizeof(regValue));
//
// We need to inform the system about this change. We do not
// tell any other apps about this (ie do not set any of the
// notification flags as the last parm)
//
SystemParametersInfo(SPI_SETLANGTOGGLE, 0, 0, 0); } else { m_caToggle = LANGUAGE_TOGGLE_NOT_PRESENT; }
RegCloseKey(hkeyToggle); } }
DebugExitVOID(ASShare::CALangToggle); }
//
// CAStartControlled()
//
void ASShare::CAStartControlled ( ASPerson * pasInControl, UINT controlID ) { DebugEntry(ASShare::CAStartControlled);
ValidatePerson(pasInControl);
//
// Undo last known state of remote
//
CAClearRemoteState(pasInControl);
//
// Get any VIEW frame UI out of the way
//
SendMessage(g_asSession.hwndHostUI, HOST_MSG_CONTROLLED, TRUE, 0); VIEWStartControlled(TRUE);
ASSERT(!m_pasLocal->m_caControlledBy); m_pasLocal->m_caControlledBy = pasInControl;
ASSERT(!pasInControl->m_caInControlOf); pasInControl->m_caInControlOf = m_pasLocal;
ASSERT(!pasInControl->m_caControlID); ASSERT(controlID); pasInControl->m_caControlID = controlID;
//
// Notify IM.
//
IM_Controlled(pasInControl);
//
// Disable language toggling.
//
CALangToggle(FALSE);
ASSERT(m_pHost); m_pHost->CM_Controlled(pasInControl);
//
// Notify the UI. Pass GCCID of controller
//
DCS_NotifyUI(SH_EVT_STARTCONTROLLED, pasInControl->cpcCaps.share.gccID, 0);
//
// Broadcast new state
//
m_pHost->m_caRetrySendState = TRUE; m_pHost->CA_Periodic();
DebugExitVOID(ASShare::CAStartControlled); }
//
// CAStopControlled()
//
void ASShare::CAStopControlled(void) { ASPerson * pasControlledBy;
DebugEntry(ASShare::CAStopControlled);
pasControlledBy = m_pasLocal->m_caControlledBy; ValidatePerson(pasControlledBy);
//
// If control is paused, unpause it.
//
if (m_pasLocal->m_caControlPaused) { CA_PauseControl(pasControlledBy, FALSE, FALSE); }
m_pasLocal->m_caControlledBy = NULL;
ASSERT(pasControlledBy->m_caInControlOf == m_pasLocal); pasControlledBy->m_caInControlOf = NULL;
ASSERT(pasControlledBy->m_caControlID); pasControlledBy->m_caControlID = 0;
//
// Notify IM.
//
IM_Controlled(NULL);
//
// Restore language toggling functionality.
//
CALangToggle(TRUE);
ASSERT(m_pHost); m_pHost->CM_Controlled(NULL);
VIEWStartControlled(FALSE); ASSERT(IsWindow(g_asSession.hwndHostUI)); SendMessage(g_asSession.hwndHostUI, HOST_MSG_CONTROLLED, FALSE, 0);
//
// Notify the UI
//
DCS_NotifyUI(SH_EVT_STOPCONTROLLED, pasControlledBy->cpcCaps.share.gccID, 0);
//
// Broadcast the new state
//
m_pHost->m_caRetrySendState = TRUE; m_pHost->CA_Periodic();
DebugExitVOID(ASShare::CAStopControlled); }
//
// CAStartInControl()
//
void ASShare::CAStartInControl ( ASPerson * pasControlled, UINT controlID ) { DebugEntry(ASShare::CAStartInControl);
ValidatePerson(pasControlled);
//
// Undo last known state of host
//
CAClearRemoteState(pasControlled);
ASSERT(!m_pasLocal->m_caInControlOf); m_pasLocal->m_caInControlOf = pasControlled;
ASSERT(!pasControlled->m_caControlledBy); pasControlled->m_caControlledBy = m_pasLocal;
ASSERT(!pasControlled->m_caControlID); ASSERT(controlID); pasControlled->m_caControlID = controlID;
ASSERT(!g_lpimSharedData->imControlled); IM_InControl(pasControlled);
VIEW_InControl(pasControlled, TRUE);
//
// Pass GCC ID of node we're controlling
//
DCS_NotifyUI(SH_EVT_STARTINCONTROL, pasControlled->cpcCaps.share.gccID, 0);
DebugExitVOID(ASShare::CAStartInControl); }
//
// CAStopInControl()
//
void ASShare::CAStopInControl(void) { ASPerson * pasInControlOf;
DebugEntry(ASShare::CAStopInControl);
pasInControlOf = m_pasLocal->m_caInControlOf; ValidatePerson(pasInControlOf);
if (pasInControlOf->m_caControlPaused) { pasInControlOf->m_caControlPaused = FALSE; }
m_pasLocal->m_caInControlOf = NULL;
ASSERT(pasInControlOf->m_caControlledBy == m_pasLocal); pasInControlOf->m_caControlledBy = NULL;
ASSERT(pasInControlOf->m_caControlID); pasInControlOf->m_caControlID = 0;
ASSERT(!g_lpimSharedData->imControlled); IM_InControl(NULL);
VIEW_InControl(pasInControlOf, FALSE);
DCS_NotifyUI(SH_EVT_STOPINCONTROL, pasInControlOf->cpcCaps.share.gccID, 0);
DebugExitVOID(ASShare::CAStopInControl); }
//
// CA_AllowControl()
// Allows/disallows remotes from controlling us.
//
void ASShare::CA_AllowControl(BOOL fAllow) { DebugEntry(ASShare::CA_AllowControl);
if (!m_pHost) { WARNING_OUT(("CA_AllowControl: ignoring, we aren't hosting")); DC_QUIT; }
if (fAllow != m_pasLocal->m_caAllowControl) { if (!fAllow) { // Undo pending control/control queries/being controlled stuff
CA_ClearLocalState(CACLEAR_HOST, NULL, TRUE); }
m_pasLocal->m_caAllowControl = fAllow;
ASSERT(IsWindow(g_asSession.hwndHostUI)); SendMessage(g_asSession.hwndHostUI, HOST_MSG_ALLOWCONTROL, fAllow, 0);
DCS_NotifyUI(SH_EVT_CONTROLLABLE, fAllow, 0);
m_pHost->m_caRetrySendState = TRUE; }
DC_EXIT_POINT: DebugExitVOID(ASShare::CA_AllowControl); }
//
// CA_HostEnded()
//
// When we stop hosting, we do not need to flush queued control
// responses. But we need to delete them!
//
void ASHost::CA_HostEnded(void) { PCAREQUEST pCARequest; PCAREQUEST pCANext;
DebugEntry(ASHost::CA_HostEnded);
m_pShare->CA_ClearLocalState(CACLEAR_HOST, NULL, FALSE);
//
// Delete now obsolete messages originating from us as host.
//
pCARequest = (PCAREQUEST)COM_BasedListFirst(&m_pShare->m_caQueuedMsgs, FIELD_OFFSET(CAREQUEST, chain)); while (pCARequest) { pCANext = (PCAREQUEST)COM_BasedListNext(&m_pShare->m_caQueuedMsgs, pCARequest, FIELD_OFFSET(CAREQUEST, chain));
if (pCARequest->type == REQUEST_30) { switch (pCARequest->msg) { //
// Delete messages sent by us when we are hosting.
//
case CA_INFORM_PAUSEDCONTROL: case CA_INFORM_UNPAUSEDCONTROL: case CA_REPLY_REQUEST_TAKECONTROL: case CA_REQUEST_GIVECONTROL: WARNING_OUT(("Deleting host control message %d, we stopped hosting", pCARequest->msg)); COM_BasedListRemove(&pCARequest->chain); delete pCARequest; break; } }
pCARequest = pCANext; }
if (m_pShare->m_pasLocal->m_caAllowControl) { m_pShare->m_pasLocal->m_caAllowControl = FALSE;
ASSERT(IsWindow(g_asSession.hwndHostUI)); SendMessage(g_asSession.hwndHostUI, HOST_MSG_ALLOWCONTROL, FALSE, 0);
DCS_NotifyUI(SH_EVT_CONTROLLABLE, FALSE, 0); }
DebugExitVOID(ASHost::CA_HostEnded); }
//
// CA_TakeControl()
//
// Called by viewer to ask to take control of host. Note parallels to
// CA_GiveControl(), which is called by host to get same result.
//
void ASShare::CA_TakeControl(ASPerson * pasHost) { DebugEntry(ASShare::CA_TakeControl);
ValidatePerson(pasHost); ASSERT(pasHost != m_pasLocal);
//
// If this person isn't hosting or controllable, fail.
//
if (!pasHost->m_pView) { WARNING_OUT(("CA_TakeControl: failing, person [%d] not hosting", pasHost->mcsID)); DC_QUIT; }
if (!pasHost->m_caAllowControl) { WARNING_OUT(("CA_TakeControl: failing, host [%d] not controllable", pasHost->mcsID)); DC_QUIT; }
//
// Undo current state.
//
CA_ClearLocalState(CACLEAR_ALL, NULL, TRUE);
//
// Now take control.
//
if (pasHost->cpcCaps.general.version >= CAPS_VERSION_30) { //
// 3.0 host
//
CA30P packetSend;
ZeroMemory(&packetSend, sizeof(packetSend)); packetSend.rtc.viewerControlID = CANewRequestID();
if (CAQueueSendPacket(pasHost->mcsID, CA_REQUEST_TAKECONTROL, &packetSend)) { //
// Now we're in waiting state.
//
CAStartWaiting(pasHost, CA_REPLY_REQUEST_TAKECONTROL); VIEW_UpdateStatus(pasHost, IDS_STATUS_WAITINGFORCONTROL); } else { WARNING_OUT(("CA_TakeControl of [%d]: failing, out of memory", pasHost->mcsID)); } } else { CA2xTakeControl(pasHost); }
DC_EXIT_POINT: DebugExitVOID(ASShare::CA_TakeControl); }
//
// CA_CancelTakeControl()
//
void ASShare::CA_CancelTakeControl ( ASPerson * pasHost, BOOL fPacket ) { DebugEntry(ASShare::CA_CancelTakeControl);
ValidatePerson(pasHost); ASSERT(pasHost != m_pasLocal);
if ((m_caWaitingForReplyFrom != pasHost) || (m_caWaitingForReplyMsg != CA_REPLY_REQUEST_TAKECONTROL)) { // We're not waiting for control of this host.
WARNING_OUT(("CA_CancelTakeControl failing; not waiting to take control of [%d]", pasHost->mcsID)); DC_QUIT; }
ASSERT(pasHost->cpcCaps.general.version >= CAPS_VERSION_30); ASSERT(pasHost->m_caControlID == 0);
if (fPacket) { CA30P packetSend;
ZeroMemory(&packetSend, sizeof(packetSend)); packetSend.inform.viewerControlID = m_pasLocal->m_caControlID; packetSend.inform.hostControlID = pasHost->m_caControlID;
if (!CAQueueSendPacket(pasHost->mcsID, CA_INFORM_RELEASEDCONTROL, &packetSend)) { WARNING_OUT(("Couldn't tell node [%d] we're no longer waiting for control", pasHost->mcsID)); } }
m_caWaitingForReplyFrom = NULL; m_caWaitingForReplyMsg = 0;
VIEW_UpdateStatus(pasHost, IDS_STATUS_NONE);
DC_EXIT_POINT: DebugExitVOID(ASShare::CA_CancelTakeControl); }
//
// CA_ReleaseControl()
//
void ASShare::CA_ReleaseControl ( ASPerson * pasHost, BOOL fPacket ) { DebugEntry(ASShare::CA_ReleaseControl);
ValidatePerson(pasHost); ASSERT(pasHost != m_pasLocal);
if (pasHost->m_caControlledBy != m_pasLocal) { // We're not in control of this dude, nothing to do.
WARNING_OUT(("CA_ReleaseControl failing; not in control of [%d]", pasHost->mcsID)); DC_QUIT; }
ASSERT(!m_caWaitingForReplyFrom); ASSERT(!m_caWaitingForReplyMsg);
if (fPacket) { if (pasHost->cpcCaps.general.version >= CAPS_VERSION_30) { CA30P packetSend;
ZeroMemory(&packetSend, sizeof(packetSend)); packetSend.inform.viewerControlID = m_pasLocal->m_caControlID; packetSend.inform.hostControlID = pasHost->m_caControlID;
if (!CAQueueSendPacket(pasHost->mcsID, CA_INFORM_RELEASEDCONTROL, &packetSend)) { WARNING_OUT(("Couldn't tell node [%d] they're no longer controlled", pasHost->mcsID)); } } else { if (!CA2xQueueSendMsg(0, CA_OLDMSG_DETACH, 0, 0)) { WARNING_OUT(("Couldn't tell 2.x node [%d] they're no longer controlled", pasHost->mcsID)); } } }
CAStopInControl();
DC_EXIT_POINT: DebugExitVOID(ASShare::CA_ReleaseControl); }
//
// CA_PassControl()
//
void ASShare::CA_PassControl(ASPerson * pasHost, ASPerson * pasViewer) { CA30P packetSend;
DebugEntry(ASShare::CA_PassControl);
ValidatePerson(pasHost); ValidatePerson(pasViewer); ASSERT(pasHost != pasViewer); ASSERT(pasHost != m_pasLocal); ASSERT(pasViewer != m_pasLocal);
if (pasHost->m_caControlledBy != m_pasLocal) { WARNING_OUT(("CA_PassControl: failing, we're not in control of [%d]", pasHost->mcsID)); DC_QUIT; }
ASSERT(!m_caWaitingForReplyFrom); ASSERT(!m_caWaitingForReplyMsg);
//
// No 2.x nodes, neither host nor controller, allowed
//
if ((pasHost->cpcCaps.general.version < CAPS_VERSION_30) || (pasViewer->cpcCaps.general.version < CAPS_VERSION_30)) { WARNING_OUT(("CA_PassControl: failing, we can't pass control with 2.x nodes")); DC_QUIT; }
ZeroMemory(&packetSend, sizeof(packetSend)); packetSend.ppc.viewerControlID = m_pasLocal->m_caControlID; packetSend.ppc.hostControlID = pasHost->m_caControlID; packetSend.ppc.mcsPassTo = pasViewer->mcsID;
if (CAQueueSendPacket(pasHost->mcsID, CA_PREFER_PASSCONTROL, &packetSend)) { CAStopInControl(); } else { WARNING_OUT(("Couldn't tell node [%d] we want them to pass control to [%d]", pasHost->mcsID, pasViewer->mcsID)); }
DC_EXIT_POINT: DebugExitVOID(ASShare::CA_PassControl); }
//
// CA_GiveControl()
//
// Called by host to ask to grant control to viewer. Note parallels to
// CA_TakeControl(), which is called by viewer to get same result.
//
void ASShare::CA_GiveControl(ASPerson * pasTo) { CA30P packetSend;
DebugEntry(ASShare::CA_GiveControl);
ValidatePerson(pasTo); ASSERT(pasTo != m_pasLocal);
//
// If we aren't hosting or controllable, fail.
//
if (!m_pHost) { WARNING_OUT(("CA_GiveControl: failing, we're not hosting")); DC_QUIT; }
if (!m_pasLocal->m_caAllowControl) { WARNING_OUT(("CA_GiveControl: failing, we're not controllable")); DC_QUIT; }
if (pasTo->cpcCaps.general.version < CAPS_VERSION_30) { //
// Can't do this with 2.x node.
//
WARNING_OUT(("CA_GiveControl: failing, can't invite 2.x node [%d]", pasTo->mcsID)); DC_QUIT; }
//
// Undo our control state.
//
CA_ClearLocalState(CACLEAR_ALL, NULL, TRUE);
//
// Now invite control.
//
ZeroMemory(&packetSend, sizeof(packetSend)); packetSend.rgc.hostControlID = CANewRequestID(); packetSend.rgc.mcsPassFrom = 0;
if (CAQueueSendPacket(pasTo->mcsID, CA_REQUEST_GIVECONTROL, &packetSend)) { //
// Now we're in waiting state.
//
CAStartWaiting(pasTo, CA_REPLY_REQUEST_GIVECONTROL); } else { WARNING_OUT(("CA_GiveControl of [%d]: failing, out of memory", pasTo->mcsID)); }
DC_EXIT_POINT: DebugExitVOID(ASShare::CA_GiveControl); }
//
// CA_CancelGiveControl()
// Cancels an invite TAKE or PASS request.
//
void ASShare::CA_CancelGiveControl ( ASPerson * pasTo, BOOL fPacket ) { DebugEntry(ASShare::CA_CancelGiveControl);
ValidatePerson(pasTo); ASSERT(pasTo != m_pasLocal);
//
// Have we invited this person, and are we now waiting for a response?
//
if ((m_caWaitingForReplyFrom != pasTo) || (m_caWaitingForReplyMsg != CA_REPLY_REQUEST_GIVECONTROL)) { // We're not waiting to be controlled by this viewer.
WARNING_OUT(("CA_CancelGiveControl failing; not waiting to give control to [%d]", pasTo->mcsID)); DC_QUIT; }
ASSERT(pasTo->cpcCaps.general.version >= CAPS_VERSION_30); ASSERT(!pasTo->m_caControlID);
if (fPacket) { CA30P packetSend;
ZeroMemory(&packetSend, sizeof(packetSend)); packetSend.inform.viewerControlID = pasTo->m_caControlID; packetSend.inform.hostControlID = m_pasLocal->m_caControlID;
if (!CAQueueSendPacket(pasTo->mcsID, CA_INFORM_REVOKEDCONTROL, &packetSend)) { WARNING_OUT(("Couldn't tell node [%d] they're no longer invited to control us", pasTo->mcsID)); } }
m_caWaitingForReplyFrom = NULL; m_caWaitingForReplyMsg = 0;
DC_EXIT_POINT: DebugExitVOID(ASShare::CA_CancelGiveControl); }
//
// CA_RevokeControl()
// Takes control back. If we're cleaning up (we've stopped hosting or
//
//
void ASShare::CA_RevokeControl ( ASPerson * pasInControl, BOOL fPacket ) { CA30P packetSend; PCAREQUEST pRequest;
DebugEntry(ASShare::CA_RevokeControl);
//
// If the response to pasController is still queued, simply delete it.
// There should NOT be any CARESULT_CONFIRMED responses left.
//
// Otherwise, if it wasn't found, we must send a packet.
//
ValidatePerson(pasInControl); ASSERT(pasInControl != m_pasLocal);
if (pasInControl != m_pasLocal->m_caControlledBy) { WARNING_OUT(("CA_RevokeControl: node [%d] not in control of us", pasInControl->mcsID)); DC_QUIT; }
//
// Take control back if we're being controlled
//
if (fPacket) { //
// Regardless of whether we can queue or not, we get control back!
// Note that we use the controller's request ID, so he knows if
// this is still applicable.
//
ZeroMemory(&packetSend, sizeof(packetSend)); packetSend.inform.viewerControlID = pasInControl->m_caControlID; packetSend.inform.hostControlID = m_pasLocal->m_caControlID;
if (!CAQueueSendPacket(pasInControl->mcsID, CA_INFORM_REVOKEDCONTROL, &packetSend))
{ WARNING_OUT(("Couldn't tell node [%d] they're no longer in control", pasInControl->mcsID)); } }
CAStopControlled();
DC_EXIT_POINT: DebugExitVOID(ASShare::CA_RevokeControl); }
//
// CA_PauseControl()
//
void ASShare::CA_PauseControl ( ASPerson * pasControlledBy, BOOL fPause, BOOL fPacket ) { DebugEntry(ASShare::CA_PauseControl);
ValidatePerson(pasControlledBy); ASSERT(pasControlledBy != m_pasLocal);
//
// If we aren't a controlled host, this doesn't do anything.
//
if (pasControlledBy != m_pasLocal->m_caControlledBy) { WARNING_OUT(("CA_PauseControl failing; not controlled by [%d]", pasControlledBy->mcsID)); DC_QUIT; }
ASSERT(m_pHost); ASSERT(m_pasLocal->m_caAllowControl);
if (m_pasLocal->m_caControlPaused == (fPause != FALSE)) { WARNING_OUT(("CA_PauseControl failing; already in requested state")); DC_QUIT; }
if (fPacket) { CA30P packetSend;
ZeroMemory(&packetSend, sizeof(packetSend)); packetSend.inform.viewerControlID = m_pasLocal->m_caControlledBy->m_caControlID; packetSend.inform.hostControlID = m_pasLocal->m_caControlID;
if (!CAQueueSendPacket(m_pasLocal->m_caControlledBy->mcsID, (fPause ? CA_INFORM_PAUSEDCONTROL : CA_INFORM_UNPAUSEDCONTROL), &packetSend)) { WARNING_OUT(("CA_PauseControl: out of memory, can't notify [%d]", m_pasLocal->m_caControlledBy->mcsID)); } }
// Do pause
m_pasLocal->m_caControlPaused = (fPause != FALSE); g_lpimSharedData->imPaused = (fPause != FALSE);
DCS_NotifyUI((fPause ? SH_EVT_PAUSEDCONTROLLED : SH_EVT_UNPAUSEDCONTROLLED), pasControlledBy->cpcCaps.share.gccID, 0);
DC_EXIT_POINT: DebugExitVOID(ASShare::CA_PauseControl); }
//
// CAHandleRequestTakeControl()
// WE are HOST, REMOTE is VIEWER
// Handles incoming take control request. If our state is good, we accept.
//
void ASShare::CAHandleRequestTakeControl ( ASPerson * pasViewer, PCA_RTC_PACKET pPacketRecv ) { UINT result = CARESULT_CONFIRMED;
DebugEntry(ASShare::CAHandleRequestTakeControl);
ValidatePerson(pasViewer);
//
// If we aren't hosting, or haven't turned allow control on, we're
// not controllable.
//
if (!m_pHost || !m_pasLocal->m_caAllowControl) { result = CARESULT_DENIED_WRONGSTATE; goto RESPOND_PACKET; }
//
// Are we doing something else right now? Waiting to hear back about
// something?
//
if (m_caWaitingForReplyFrom) { result = CARESULT_DENIED_BUSY; goto RESPOND_PACKET; }
if (m_caQueryDlg) { result = CARESULT_DENIED_BUSY; goto RESPOND_PACKET; }
//
// LAURABU TEMPORARY:
// In a bit, if we're controlled when a new control request comes in,
// pause control then allow host to handle it.
//
if (m_pasLocal->m_caControlledBy) { result = CARESULT_DENIED_BUSY; goto RESPOND_PACKET; }
//
// Try to put up query dialog
//
if (!CAStartQuery(pasViewer, CA_REQUEST_TAKECONTROL, (PCA30P)pPacketRecv)) { result = CARESULT_DENIED; }
RESPOND_PACKET: if (result != CARESULT_CONFIRMED) { // Instant failure.
CACompleteRequestTakeControl(pasViewer, pPacketRecv, result); } else { //
// We're in a waiting state. CACompleteRequestTakeControl() will
// complete later or the request will just go away.
//
}
DebugExitVOID(ASShare::CAHandleRequestTakeControl); }
//
// CACompleteRequestTakeControl()
// WE are HOST, REMOTE is VIEWER
// Completes the take control request.
//
void ASShare::CACompleteRequestTakeControl ( ASPerson * pasFrom, PCA_RTC_PACKET pPacketRecv, UINT result ) { CA30P packetSend;
DebugEntry(ASShare::CACompleteRequestTakeControl);
ValidatePerson(pasFrom);
ZeroMemory(&packetSend, sizeof(packetSend)); packetSend.rrtc.viewerControlID = pPacketRecv->viewerControlID; packetSend.rrtc.result = result;
if (result == CARESULT_CONFIRMED) { packetSend.rrtc.hostControlID = CANewRequestID(); }
if (CAQueueSendPacket(pasFrom->mcsID, CA_REPLY_REQUEST_TAKECONTROL, &packetSend)) { if (result == CARESULT_CONFIRMED) { // Clear current state, whatever that is.
CA_ClearLocalState(CACLEAR_ALL, NULL, TRUE);
// We are now controlled by the sender.
CAStartControlled(pasFrom, pPacketRecv->viewerControlID); } else { WARNING_OUT(("Denying REQUEST TAKE CONTROL from [%d] with reason %d", pasFrom->mcsID, result)); } } else { WARNING_OUT(("Reply to REQUEST TAKE CONTROL from [%d] failing, out of memory", pasFrom->mcsID)); }
DebugExitVOID(ASShare::CACompleteRequestTakeControl); }
//
// CAHandleReplyRequestTakeControl()
// WE are VIEWER, REMOTE is HOST
// Handles reply to previous take control request.
//
void ASShare::CAHandleReplyRequestTakeControl ( ASPerson * pasHost, PCA_REPLY_RTC_PACKET pPacketRecv ) { DebugEntry(ASShare::CAHandleReplyRequestTakeControl);
ValidatePerson(pasHost);
if (pPacketRecv->result == CARESULT_CONFIRMED) { // On success, should have valid op ID.
ASSERT(pPacketRecv->hostControlID); } else { // On failure, should have invalid op ID.
ASSERT(!pPacketRecv->hostControlID); }
//
// Is this response for the current control op?
//
if ((m_caWaitingForReplyFrom != pasHost) || (m_caWaitingForReplyMsg != CA_REPLY_REQUEST_TAKECONTROL)) { WARNING_OUT(("Ignoring TAKE CONTROL REPLY from [%d], not waiting for one", pasHost->mcsID)); DC_QUIT; }
if (pPacketRecv->viewerControlID != m_pasLocal->m_caControlID) { WARNING_OUT(("Ignoring TAKE CONTROL REPLY from [%d], request %d is out of date", pasHost->mcsID, pPacketRecv->viewerControlID)); DC_QUIT;
}
ASSERT(!m_caQueryDlg);
//
// Cleanup waiting state (for both failure & success)
//
CA_CancelTakeControl(pasHost, FALSE); ASSERT(!m_caWaitingForReplyFrom); ASSERT(!m_caWaitingForReplyMsg);
if (pPacketRecv->result == CARESULT_CONFIRMED) { // Success! We're now in control of the host.
// Make sure our own state is OK
ASSERT(!m_pasLocal->m_caControlledBy); ASSERT(!m_pasLocal->m_caInControlOf);
CAStartInControl(pasHost, pPacketRecv->hostControlID); } else { UINT ids;
WARNING_OUT(("TAKE CONTROL REPLY from host [%d] is failure %d", pasHost->mcsID, pPacketRecv->result));
ids = IDS_ERR_TAKECONTROL_MIN + pPacketRecv->result; if ((ids < IDS_ERR_TAKECONTROL_FIRST) || (ids > IDS_ERR_TAKECONTROL_LAST)) ids = IDS_ERR_TAKECONTROL_LAST;
VIEW_Message(pasHost, ids); }
DC_EXIT_POINT: DebugExitVOID(ASShare::CAHandleReplyRequestTakeControl); }
//
// CAHandleRequestGiveControl()
// WE are VIEWER, REMOTE is HOST
// Handles incoming take control invite. If our state is good, we accept.
//
// NOTE how similar this routine is to CAHandleRequestTakeControl(). They
// are inverses of each other. With RequestTake/Reply sequence, viewer
// initiates, host finishes. With RequestGive/Reply sequence, host initiates,
// viewer finishes. Both end up with viewer in control of host when
// completed successfully.
//
void ASShare::CAHandleRequestGiveControl ( ASPerson * pasHost, PCA_RGC_PACKET pPacketRecv ) { UINT result = CARESULT_CONFIRMED;
DebugEntry(ASShare::CAHandleRequestGiveControl);
ValidatePerson(pasHost);
//
// Is this node hosting as far as we know. If not, or has not turned
// on allow control, we can't do it.
//
if (!pasHost->m_pView) { WARNING_OUT(("GIVE CONTROL went ahead of HOSTING, that's bad")); result = CARESULT_DENIED_WRONGSTATE; goto RESPOND_PACKET; }
if (!pasHost->m_caAllowControl) { //
// We haven't got an AllowControl notification yet, this info is
// more up to-date. Make use of it.
//
WARNING_OUT(("GIVE CONTROL went ahead of ALLOW CONTROL, that's kind of bad")); result = CARESULT_DENIED_WRONGSTATE; goto RESPOND_PACKET; }
//
// Are we doing something else right now? Waiting to hear back about
// something?
//
if (m_caWaitingForReplyFrom) { result = CARESULT_DENIED_BUSY; goto RESPOND_PACKET; }
if (m_caQueryDlg) { result = CARESULT_DENIED_BUSY; goto RESPOND_PACKET; }
//
// LAURABU TEMPORARY:
// In a bit, if we're controlled when a new control request comes in,
// pause control then allow host to handle it.
//
if (m_pasLocal->m_caControlledBy) { result = CARESULT_DENIED_BUSY; goto RESPOND_PACKET; }
//
// Try to put up query dialog
//
if (!CAStartQuery(pasHost, CA_REQUEST_GIVECONTROL, (PCA30P)pPacketRecv)) { result = CARESULT_DENIED; }
RESPOND_PACKET: if (result != CARESULT_CONFIRMED) { // Instant failure.
CACompleteRequestGiveControl(pasHost, pPacketRecv, result); } else { //
// We're in a waiting state. CACompleteRequestGiveControl() will
// complete later or the request will just go away.
//
}
DebugExitVOID(ASShare::CAHandleRequestGiveControl); }
//
// CACompleteRequestGiveControl()
// WE are VIEWER, REMOTE is HOST
// Completes the invite control request.
//
void ASShare::CACompleteRequestGiveControl ( ASPerson * pasFrom, PCA_RGC_PACKET pPacketRecv, UINT result ) { CA30P packetSend;
DebugEntry(ASShare::CACompleteRequestGiveControl);
ValidatePerson(pasFrom);
ZeroMemory(&packetSend, sizeof(packetSend)); packetSend.rrgc.hostControlID = pPacketRecv->hostControlID; packetSend.rrgc.result = result;
if (result == CARESULT_CONFIRMED) { packetSend.rrgc.viewerControlID = CANewRequestID(); }
if (CAQueueSendPacket(pasFrom->mcsID, CA_REPLY_REQUEST_GIVECONTROL, &packetSend)) { //
// If this is successful, change our state. We're now in control.
//
if (result == CARESULT_CONFIRMED) { // Clear current state, whatever that is.
CA_ClearLocalState(CACLEAR_ALL, NULL, TRUE);
CAStartInControl(pasFrom, pPacketRecv->hostControlID); } else { WARNING_OUT(("Denying GIVE CONTROL from [%d] with reason %d", pasFrom->mcsID, result)); } } else { WARNING_OUT(("Reply to GIVE CONTROL from [%d] failing, out of memory", pasFrom->mcsID)); }
DebugExitVOID(ASShare::CACompleteRequestGiveControl); }
//
// CAHandleReplyRequestGiveControl()
// WE are HOST, REMOTE is VIEWER
// Handles reply to previous take control invite.
//
void ASShare::CAHandleReplyRequestGiveControl ( ASPerson * pasViewer, PCA_REPLY_RGC_PACKET pPacketRecv ) { DebugEntry(ASShare::CAHandleReplyRequestGiveControl);
ValidatePerson(pasViewer);
if (pPacketRecv->result == CARESULT_CONFIRMED) { // On success, should have valid op ID.
ASSERT(pPacketRecv->viewerControlID); } else { // On failure, should have invalid op ID.
ASSERT(!pPacketRecv->viewerControlID); }
//
// Is this response for the latest control op?
//
if ((m_caWaitingForReplyFrom != pasViewer) || (m_caWaitingForReplyMsg != CA_REPLY_REQUEST_GIVECONTROL)) { WARNING_OUT(("Ignoring GIVE CONTROL REPLY from [%d], not waiting for one", pasViewer->mcsID)); DC_QUIT; }
if (pPacketRecv->hostControlID != m_pasLocal->m_caControlID) { WARNING_OUT(("Ignoring GIVE CONTROL REPLY from [%d], request %d is out of date", pasViewer->mcsID, pPacketRecv->hostControlID)); DC_QUIT; }
ASSERT(!m_caQueryDlg); ASSERT(m_pHost); ASSERT(m_pasLocal->m_caAllowControl);
//
// Cleanup waiting state (for both failure & success)
//
CA_CancelGiveControl(pasViewer, FALSE); ASSERT(!m_caWaitingForReplyFrom); ASSERT(!m_caWaitingForReplyMsg);
if (pPacketRecv->result == CARESULT_CONFIRMED) { // Success! We are now controlled by the viewer
// Make sure our own state is OK
ASSERT(!m_pasLocal->m_caControlledBy); ASSERT(!m_pasLocal->m_caInControlOf);
CAStartControlled(pasViewer, pPacketRecv->viewerControlID); } else { WARNING_OUT(("GIVE CONTROL to viewer [%d] was denied", pasViewer->mcsID)); }
DC_EXIT_POINT: DebugExitVOID(ASShare::CAHandleReplyRequestGiveControl); }
//
// CAHandlePreferPassControl()
// WE are HOST, REMOTE is CONTROLLER
// Handles incoming pass control request. If we are controlled by the
// remote, and end user is cool with it, accept.
//
void ASShare::CAHandlePreferPassControl ( ASPerson * pasController, PCA_PPC_PACKET pPacketRecv ) { ASPerson * pasNewController;
DebugEntry(ASShare::CAHandlePreferPassControl);
ValidatePerson(pasController);
//
// If we're not controlled by the requester, ignore it.
//
if (m_pasLocal->m_caControlledBy != pasController) { WARNING_OUT(("Ignoring PASS CONTROL from [%d], not controlled by him", pasController->mcsID)); DC_QUIT; }
if ((pPacketRecv->viewerControlID != pasController->m_caControlID) || (pPacketRecv->hostControlID != m_pasLocal->m_caControlID)) { WARNING_OUT(("Ignoring PASS CONTROL from [%d], request %d %d out of date", pasController->mcsID, pPacketRecv->viewerControlID, pPacketRecv->hostControlID)); DC_QUIT; }
ASSERT(!m_caQueryDlg); ASSERT(!m_caWaitingForReplyFrom); ASSERT(!m_caWaitingForReplyMsg);
//
// OK, the sender is not in control of us anymore.
//
CA_RevokeControl(pasController, FALSE);
// Is the pass to person specified valid?
pasNewController = SC_PersonFromNetID(pPacketRecv->mcsPassTo); if (!pasNewController || (pasNewController == pasController) || (pasNewController == m_pasLocal) || (pasNewController->cpcCaps.general.version < CAPS_VERSION_30)) { WARNING_OUT(("PASS CONTROL to [%d] failing, not valid person to pass to", pPacketRecv->mcsPassTo)); DC_QUIT; }
//
// Try to put up query dialog
//
if (!CAStartQuery(pasController, CA_PREFER_PASSCONTROL, (PCA30P)pPacketRecv)) { // Instant failure. In this case, no packet.
WARNING_OUT(("Denying PREFER PASS CONTROL from [%d], out of memory", pasController->mcsID)); } else { //
// We're in a waiting state. CACompletePreferPassControl() will
// complete later or the request will just go away.
//
}
DC_EXIT_POINT: DebugExitVOID(ASShare::CAHandlePreferPassControl); }
//
// CACompletePreferPassControl()
// WE are HOST, REMOTE is new potential CONTROLLER
// Completes the prefer pass control request.
//
void ASShare::CACompletePreferPassControl ( ASPerson * pasTo, UINT_PTR mcsOrg, PCA_PPC_PACKET pPacketRecv, UINT result ) { CA30P packetSend;
DebugEntry(ASShare::CACompletePreferPassControl);
ValidatePerson(pasTo);
if (result == CARESULT_CONFIRMED) { ZeroMemory(&packetSend, sizeof(packetSend)); packetSend.rgc.hostControlID = CANewRequestID(); packetSend.rgc.mcsPassFrom = mcsOrg;
if (CAQueueSendPacket(pasTo->mcsID, CA_REQUEST_GIVECONTROL, &packetSend)) { CA_ClearLocalState(CACLEAR_HOST, NULL, TRUE);
CAStartWaiting(pasTo, CA_REPLY_REQUEST_GIVECONTROL); } else { WARNING_OUT(("Reply to PREFER PASS CONTROL from [%d] to [%d] failing, out of memory", mcsOrg, pasTo->mcsID)); } } else { WARNING_OUT(("Denying PREFER PASS CONTROL from [%d] to [%d] with reason %d", mcsOrg, pasTo->mcsID, result)); }
DebugExitVOID(ASShare::CACompletePreferPassControl); }
//
// CAHandleInformReleasedControl()
// WE are HOST, REMOTE is CONTROLLER
//
void ASShare::CAHandleInformReleasedControl ( ASPerson * pasController, PCA_INFORM_PACKET pPacketRecv ) { DebugEntry(ASShare::CAHandleInformReleasedControl);
ValidatePerson(pasController);
//
// Do we currently have a TakeControl dialog up for this request? If so,
// take it down but don't send a packet.
//
if (m_caQueryDlg && (m_caQuery.pasReplyTo == pasController) && (m_caQuery.msg == CA_REQUEST_TAKECONTROL) && (m_caQuery.request.rtc.viewerControlID == pPacketRecv->viewerControlID)) { ASSERT(!pPacketRecv->hostControlID); CACancelQuery(pasController, FALSE); DC_QUIT; }
//
// If this person isn't in control of us or the control op referred to
// isn't the current one, ignore. NULL hostControlID means the person
// cancelled a request before they heard back from us.
//
if (pasController->m_caInControlOf != m_pasLocal) { WARNING_OUT(("Ignoring RELEASE CONTROL from [%d], we're not controlled by them", pasController->mcsID)); DC_QUIT; }
if (pPacketRecv->viewerControlID != pasController->m_caControlID) { WARNING_OUT(("Ignoring RELEASE CONTROL from [%d], viewer ID out of date", pasController->mcsID, pPacketRecv->viewerControlID)); DC_QUIT; }
if (pPacketRecv->hostControlID && (pPacketRecv->hostControlID != m_pasLocal->m_caControlID)) { WARNING_OUT(("Ignoring RELEASE CONTROL from [%d], host ID out of date", pasController->mcsID, pPacketRecv->hostControlID)); DC_QUIT; }
// Undo control, but no packet gets sent, we're just cleaning up.
CA_RevokeControl(pasController, FALSE);
DC_EXIT_POINT: DebugExitVOID(ASShare::CAHandleInformReleasedControl); }
//
// CAHandleInformRevokedControl()
// WE are CONTROLLER, REMOTE is HOST
//
void ASShare::CAHandleInformRevokedControl ( ASPerson * pasHost, PCA_INFORM_PACKET pPacketRecv ) { DebugEntry(ASShare::CAHandleInformRevokedControl);
ValidatePerson(pasHost);
//
// Do we currently have a GiveControl dialog up for this request? If so,
// take it down but don't send a packet.
//
if (m_caQueryDlg && (m_caQuery.pasReplyTo == pasHost) && (m_caQuery.msg == CA_REQUEST_GIVECONTROL) && (m_caQuery.request.rgc.hostControlID == pPacketRecv->hostControlID)) { ASSERT(!pPacketRecv->viewerControlID); CACancelQuery(pasHost, FALSE); DC_QUIT; }
//
// If this person isn't controlled by us or the control op referred to
// isn't the current one, ignore.
//
if (pasHost->m_caControlledBy != m_pasLocal) { WARNING_OUT(("Ignoring REVOKE CONTROL from [%d], not in control of them", pasHost->mcsID)); DC_QUIT; }
if (pPacketRecv->hostControlID != pasHost->m_caControlID) { WARNING_OUT(("Ignoring REVOKE CONTROL from [%d], host ID out of date", pasHost->mcsID, pPacketRecv->hostControlID)); DC_QUIT; }
if (pPacketRecv->viewerControlID && (pPacketRecv->viewerControlID != m_pasLocal->m_caControlID)) { WARNING_OUT(("Ignoring REVOKE CONTROL from [%d], viewer ID out of date", pasHost->mcsID, pPacketRecv->viewerControlID)); DC_QUIT; }
// Undo control, but no packet gets sent, we're just cleaning up.
CA_ReleaseControl(pasHost, FALSE);
DC_EXIT_POINT: DebugExitVOID(ASShare::CAHandleInformRevokedControl); }
//
// CAHandleInformPausedControl()
// WE are CONTROLLER, REMOTE is HOST
//
void ASShare::CAHandleInformPausedControl ( ASPerson * pasHost, PCA_INFORM_PACKET pPacketRecv ) { DebugEntry(ASShare::CAHandleInformPausedControl);
ValidatePerson(pasHost);
if (pasHost->m_caControlledBy != m_pasLocal) { WARNING_OUT(("Ignoring control paused from [%d], not controlled by us", pasHost->mcsID)); DC_QUIT; }
if (pasHost->m_caControlPaused) { WARNING_OUT(("Ignoring control paused from [%d], already paused", pasHost->mcsID)); DC_QUIT; }
pasHost->m_caControlPaused = TRUE; VIEW_PausedInControl(pasHost, TRUE);
DCS_NotifyUI(SH_EVT_PAUSEDINCONTROL, pasHost->cpcCaps.share.gccID, 0);
DC_EXIT_POINT: DebugExitVOID(ASShare::CAHandleInformPausedControl); }
//
// CAHandleInformUnpausedControl()
// WE are CONTROLLER, REMOTE is HOST
//
void ASShare::CAHandleInformUnpausedControl ( ASPerson * pasHost, PCA_INFORM_PACKET pPacketRecv ) { DebugEntry(ASShare::CAHandleInformUnpausedControl);
ValidatePerson(pasHost);
if (pasHost->m_caControlledBy != m_pasLocal) { WARNING_OUT(("Ignoring control unpaused from [%d], not controlled by us", pasHost->mcsID)); DC_QUIT; }
if (!pasHost->m_caControlPaused) { WARNING_OUT(("Ignoring control unpaused from [%d], not paused", pasHost->mcsID)); DC_QUIT; }
pasHost->m_caControlPaused = FALSE; VIEW_PausedInControl(pasHost, FALSE);
DCS_NotifyUI(SH_EVT_UNPAUSEDINCONTROL, pasHost->cpcCaps.share.gccID, 0);
DC_EXIT_POINT: DebugExitVOID(ASShare::CAHandleInformUnpausedControl); }
void ASShare::CAHandleNewState ( ASPerson * pasHost, PCANOTPACKET pPacket ) { BOOL caOldAllowControl; BOOL caNewAllowControl; ASPerson * pasController;
DebugEntry(ASShare::CAHandleNewState);
//
// If this node isn't hosting, ignore this.
//
ValidatePerson(pasHost); ASSERT(pasHost->cpcCaps.general.version >= CAPS_VERSION_30); ASSERT(pasHost->hetCount);
//
// Update controllable state FIRST, so view window changes will
// reflect it.
//
caOldAllowControl = pasHost->m_caAllowControl; caNewAllowControl = ((pPacket->state & CASTATE_ALLOWCONTROL) != 0);
if (!caNewAllowControl && (pasHost->m_caControlledBy == m_pasLocal)) { //
// Fix up bogus notification
//
ERROR_OUT(("CA_STATE notification error! We're in control of [%d] but he says he's not controllable.", pasHost->mcsID)); CA_ReleaseControl(pasHost, FALSE); }
pasHost->m_caAllowControl = caNewAllowControl;
// Update/clear controller
if (!pPacket->controllerID) { pasController = NULL; } else { pasController = SC_PersonFromNetID(pPacket->controllerID); if (pasController == pasHost) { ERROR_OUT(("Bogus controller, same as host [%d]", pPacket->controllerID)); pasController = NULL; } }
if (!CAClearHostState(pasHost, pasController)) { // This failed. Put back old controllable state.
pasHost->m_caAllowControl = caOldAllowControl; }
// Force a state change if the allow state has altered
if (caOldAllowControl != pasHost->m_caAllowControl) { VIEW_HostStateChange(pasHost); }
DebugExitVOID(ASShare::CAHandleNewState); }
//
// CAStartWaiting()
// Sets up vars for waiting state.
//
void ASShare::CAStartWaiting ( ASPerson * pasWaitForReplyFrom, UINT msgWaitForReplyFrom ) { DebugEntry(ASShare::CAStartWaiting);
ValidatePerson(pasWaitForReplyFrom); ASSERT(msgWaitForReplyFrom);
ASSERT(!m_caWaitingForReplyFrom); ASSERT(!m_caWaitingForReplyMsg);
m_caWaitingForReplyFrom = pasWaitForReplyFrom; m_caWaitingForReplyMsg = msgWaitForReplyFrom;
DebugExitVOID(ASShare::CAStartWaiting); }
//
// CA_ClearLocalState()
//
// Called to reset control state for LOCAL dude.
//
void ASShare::CA_ClearLocalState ( UINT flags, ASPerson * pasRemote, BOOL fPacket ) { DebugEntry(ASShare::CA_ClearLocalState);
//
// Clear HOST stuff
//
if (flags & CACLEAR_HOST) { if (m_caWaitingForReplyMsg == CA_REPLY_REQUEST_GIVECONTROL) { if (!pasRemote || (pasRemote == m_caWaitingForReplyFrom)) { // Kill the outstanding invitation to the remote
CA_CancelGiveControl(m_caWaitingForReplyFrom, fPacket); } }
if (m_caQueryDlg && ((m_caQuery.msg == CA_REQUEST_TAKECONTROL) || (m_caQuery.msg == CA_PREFER_PASSCONTROL))) { if (!pasRemote || (pasRemote == m_caQuery.pasReplyTo)) { // Kill the user query dialog that's up
CACancelQuery(m_caQuery.pasReplyTo, fPacket); } }
if (m_pasLocal->m_caControlledBy) { if (!pasRemote || (pasRemote == m_pasLocal->m_caControlledBy)) { CA_RevokeControl(m_pasLocal->m_caControlledBy, fPacket); ASSERT(!m_pasLocal->m_caControlledBy); } } }
//
// Clear VIEW stuff
//
if (flags & CACLEAR_VIEW) { if (m_caWaitingForReplyMsg == CA_REPLY_REQUEST_TAKECONTROL) { if (!pasRemote || (pasRemote == m_caWaitingForReplyFrom)) { CA_CancelTakeControl(m_caWaitingForReplyFrom, fPacket); } }
if (m_caQueryDlg && (m_caQuery.msg == CA_REQUEST_GIVECONTROL)) { if (!pasRemote || (pasRemote == m_caQuery.pasReplyTo)) { // Kill the user query dialog that's up
CACancelQuery(m_caQuery.pasReplyTo, fPacket); } }
if (m_pasLocal->m_caInControlOf) { if (!pasRemote || (pasRemote == m_pasLocal->m_caInControlOf)) { CA_ReleaseControl(m_pasLocal->m_caInControlOf, fPacket); ASSERT(!m_pasLocal->m_caInControlOf); } } }
DebugExitVOID(ASShare::CA_ClearLocalState); }
//
// CAClearRemoteState()
//
// Called to reset all control state for a REMOTE node
//
void ASShare::CAClearRemoteState(ASPerson * pasClear) { DebugEntry(ASShare::CAClearRemoteState);
if (pasClear->m_caInControlOf) { CAClearHostState(pasClear->m_caInControlOf, NULL); ASSERT(!pasClear->m_caInControlOf); ASSERT(!pasClear->m_caControlledBy); } else if (pasClear->m_caControlledBy) { CAClearHostState(pasClear, NULL); ASSERT(!pasClear->m_caControlledBy); ASSERT(!pasClear->m_caInControlOf); }
DebugExitVOID(ASShare:CAClearRemoteState); }
//
// CAClearHostState()
//
// Called to clean up the mutual pointers when undoing a node's host state.
// We need to undo the previous states:
// * Clear the previous controller of the host
// * Clear the previous controller of the controller
// * Clear the previous controllee of the controller
//
// This may be recursive.
//
// It returns TRUE if the change takes effect, FALSE if it's ignored because
// it involves us and we have more recent information.
//
BOOL ASShare::CAClearHostState ( ASPerson * pasHost, ASPerson * pasController ) { BOOL rc = FALSE; UINT gccID;
DebugEntry(ASShare::CAClearHostState);
ValidatePerson(pasHost);
//
// If nothing is changing, do nothing
//
if (pasHost->m_caControlledBy == pasController) { TRACE_OUT(("Ignoring control change; nothing's changing")); rc = TRUE; DC_QUIT; }
//
// If the host is us, ignore.
// Also, if the host isn't hosting yet we got an in control change,
// ignore it too.
//
if ((pasHost == m_pasLocal) || (pasController && !pasHost->hetCount)) { WARNING_OUT(("Ignoring control change; host is us or not sharing")); DC_QUIT; }
//
// UNDO any old state of the controller
//
if (pasController) { if (pasController == m_pasLocal) { TRACE_OUT(("Ignoring control with us as controller")); DC_QUIT; } else if (pasController->m_caInControlOf) { ASSERT(!pasController->m_caControlledBy); ASSERT(pasController->m_caInControlOf->m_caControlledBy == pasController); rc = CAClearHostState(pasController->m_caInControlOf, NULL); if (!rc) { DC_QUIT; } ASSERT(!pasController->m_caInControlOf); } else if (pasController->m_caControlledBy) { ASSERT(!pasController->m_caInControlOf); ASSERT(pasController->m_caControlledBy->m_caInControlOf == pasController); rc = CAClearHostState(pasController, NULL); if (!rc) { DC_QUIT; } ASSERT(!pasController->m_caControlledBy); } }
//
// UNDO any old IN CONTROL state of the host
//
if (pasHost->m_caInControlOf) { ASSERT(!pasHost->m_caControlledBy); ASSERT(pasHost->m_caInControlOf->m_caControlledBy == pasHost); rc = CAClearHostState(pasHost->m_caInControlOf, NULL); if (!rc) { DC_QUIT; } ASSERT(!pasHost->m_caInControlOf); }
//
// FINALLY! Update CONTROLLED BY state of the host
//
// Clear OLD ControlledBy
if (pasHost->m_caControlledBy) { ASSERT(pasHost->m_caControlledBy->m_caInControlOf == pasHost); pasHost->m_caControlledBy->m_caInControlOf = NULL; }
// Set NEW ControlledBy
pasHost->m_caControlledBy = pasController; if (pasController) { pasController->m_caInControlOf = pasHost; gccID = pasController->cpcCaps.share.gccID; } else { gccID = 0; }
VIEW_HostStateChange(pasHost);
//
// The hosts' controller has changed. Repaint the shadow cursor with/wo
// the new initials.
//
CM_UpdateShadowCursor(pasHost, pasHost->cmShadowOff, pasHost->cmPos.x, pasHost->cmPos.y, pasHost->cmHotSpot.x, pasHost->cmHotSpot.y);
rc = TRUE;
DC_EXIT_POINT: DebugExitBOOL(ASShare::CAClearHostState, rc); return(rc); }
//
// 2.X COMPATIBILITY STUFF
// This is so that we can do a decent job of reflecting old 2.x control
// stuff, and allow a 3.0 node to take control of a 2.x system.
//
//
// CA2xCooperateChange()
//
// This is called when a 2.x node is cooperating or not. When a 2.x node
// is a host and cooperating, he is "controllable" by 3.0 standards. So
// when he starts/stops hosting or starts/stops cooperating we must
// recalculate "AllowControl"
//
void ASShare::CA2xCooperateChange ( ASPerson * pasPerson, BOOL fCooperating ) { BOOL fAllowControl;
DebugEntry(ASShare::CA2xCooperateChange);
ValidatePerson(pasPerson);
//
// If this isn't a back level system, ignore it.
//
if (pasPerson->cpcCaps.general.version >= CAPS_VERSION_30) { WARNING_OUT(("Received old CA cooperate message from 3.0 node [%d]", pasPerson->mcsID)); DC_QUIT; }
//
// Update the cooperating state.
//
pasPerson->m_ca2xCooperating = fCooperating;
//
// If cooperating & this person owns the control token, this person
// is now in control of all 2.x cooperating nodes. If we were
// controlling a 2.x host, act like we've been bounced. But we MUST
// send a packet.
//
if (fCooperating) { if (pasPerson == m_ca2xControlTokenOwner) { //
// This person is now "in control" of the 2.x cooperating nodes.
// If we were in control of a 2.x host, we've basically been
// bounced and another 2.x node is running the show. With 3.0,
// it doesn't matter and we don't need to find out what's going
// on with a 3.0 node in control of 2.x dudes.
//
if (m_pasLocal->m_caInControlOf && (m_pasLocal->m_caInControlOf->cpcCaps.general.version < CAPS_VERSION_30)) { CA_ReleaseControl(pasPerson, TRUE); } } }
//
// Figure out whether we need to set/clear AllowControl
//
fAllowControl = (fCooperating && pasPerson->m_pView);
if (pasPerson->m_caAllowControl != fAllowControl) { if (pasPerson->m_pView && !fAllowControl) { //
// This 2.x node is hosting, and no longer is cooperating.
// Cleanup the controller
//
if (pasPerson->m_caControlledBy == m_pasLocal) { CA_ReleaseControl(pasPerson, TRUE); } else { CAClearHostState(pasPerson, NULL); } }
pasPerson->m_caAllowControl = fAllowControl;
// This will do nothing if this person isn't hosting.
VIEW_HostStateChange(pasPerson); }
DC_EXIT_POINT: DebugExitVOID(ASShare::CA2xCooperateChange); }
//
// CA2xRequestControl()
//
// Called when a 2.x node requests control.
//
void ASShare::CA2xRequestControl ( ASPerson * pasPerson, PCAPACKET pCAPacket ) { DebugEntry(ASShare::CA2xRequestControl);
//
// A 2.x node has sent this. 3.0 hosts never request, they simply
// grab control.
//
ValidatePerson(pasPerson);
//
// If it's from a 3.0 node, it's an error.
//
if (pasPerson->cpcCaps.general.version >= CAPS_VERSION_30) { ERROR_OUT(("Received CA_OLDMSG_REQUEST_CONTROL from 3.0 node [%d]", pasPerson->mcsID)); DC_QUIT; }
//
// If we have the token, grant it. We must release control of a host if
// that person is 2.x.
//
if (m_ca2xControlTokenOwner == m_pasLocal) { //
// In this case, we do NOT want a dest ID. This isn't us trying to
// take control of a 2.x host. It is simply granting control to
// a 2.x dude.
//
if (CA2xQueueSendMsg(0, CA_OLDMSG_GRANTED_CONTROL, pasPerson->mcsID, m_ca2xControlGeneration)) { m_ca2xControlTokenOwner = pasPerson;
// Release control of 2.x host.
if (m_pasLocal->m_caInControlOf && (m_pasLocal->m_caInControlOf->cpcCaps.general.version < CAPS_VERSION_30)) { CA_ReleaseControl(m_pasLocal->m_caInControlOf, TRUE); } } else { ERROR_OUT(("CA2xRequestControl: Unable to respond GRANTED to node [%d]", pasPerson->mcsID)); } }
DC_EXIT_POINT: DebugExitVOID(ASShare::CA2xRequestControl); }
//
// CA2xGrantedControl()
//
// Called when any node (2.x or 3.0 controlling 2.x) broadcasts granted
// control. If we are controlling a 2.x host, it is now nuked.
//
void ASShare::CA2xGrantedControl ( ASPerson * pasPerson, PCAPACKET pCAPacket ) { DebugEntry(ASShare::CA2xGrantedControl);
ValidatePerson(pasPerson);
if ((pCAPacket->data2 >= m_ca2xControlGeneration) || ((m_ca2xControlGeneration - pCAPacket->data2) > 0x80000000)) { ASPerson * pas2xNewTokenOwner;
//
// This dude is now the controller of 2.x nodes. Remember it for
// later COOPERATE msgs. If nothing has changed (this is a sync
// broadcast for example, do nothing ourselvs).
//
pas2xNewTokenOwner = SC_PersonFromNetID(pCAPacket->data1); if (pas2xNewTokenOwner != m_ca2xControlTokenOwner) { m_ca2xControlTokenOwner = pas2xNewTokenOwner; m_ca2xControlGeneration = pCAPacket->data2;
//
// Are we in control of a 2.x node? If so, undo it.
//
if (m_pasLocal->m_caInControlOf && (m_pasLocal->m_caInControlOf->cpcCaps.general.version < CAPS_VERSION_30)) { CA_ReleaseControl(m_pasLocal->m_caInControlOf, TRUE); } } }
DebugExitVOID(ASShare::CA2xGrantedControl); }
//
// CA2xTakeControl()
//
// This fakes up packets to take control of a 2.x node. We don't broadcast,
// we send them privately just to the individual node so we don't control
// any other host but him.
//
// We do this by sending COOPERATE then GRANTED_CONTROL. If there's a
// collision, we'll see a GRANTED_CONTROL from somebody else that outdates
// ours.
//
void ASShare::CA2xTakeControl(ASPerson * pasHost) { UINT_PTR caNew2xControlGeneration;
DebugEntry(ASShare::CA2xTakeControl);
ValidateView(pasHost);
caNew2xControlGeneration = m_ca2xControlGeneration + m_pasLocal->mcsID;
if (CA2xQueueSendMsg(0, CA_OLDMSG_COOPERATE, 0, 0)) { if (!CA2xQueueSendMsg(pasHost->mcsID, CA_OLDMSG_GRANTED_CONTROL, m_pasLocal->mcsID, caNew2xControlGeneration)) { //
// Failure. Best we can do is follow it with a DETACH
//
ERROR_OUT(("CA2xTakeControl: Can't take control of [%d]", pasHost->mcsID)); CA2xQueueSendMsg(0, CA_OLDMSG_DETACH, 0, 0); } else { m_ca2xControlGeneration = caNew2xControlGeneration; m_ca2xControlTokenOwner = m_pasLocal;
CANewRequestID(); CAStartInControl(pasHost, 1); } } else { ERROR_OUT(("CA2xTakeControl: Can't take control of [%d]", pasHost->mcsID)); }
DebugExitVOID(ASShare::CA2xTakeControl); }
//
// CA2xSendMsg()
// This sends a 2.x node CA message. It returns FALSE if it can't alloc
// a packet.
//
BOOL ASShare::CA2xSendMsg ( UINT_PTR destID, UINT msg, UINT_PTR data1, UINT_PTR data2 ) { BOOL fSent = FALSE; PCAPACKET pPacket; #ifdef _DEBUG
UINT sentSize; #endif // _DEBUG
DebugEntry(ASShare::CASendPacket);
//
// For cooperate/detach, there's no target. We broadcast them no
// matter what so everybody knows what state we're in.
//
if (msg != CA_OLDMSG_GRANTED_CONTROL) { ASSERT(!destID); }
//
// WE MUST USE PROT_STR_MISC! Backlevel nodes will uncompress it
// using that prot dictionary. And note that we must broadcast 2.x
// CA packets so everybody knows what's going on.
//
pPacket = (PCAPACKET)SC_AllocPkt(PROT_STR_MISC, g_s20BroadcastID, sizeof(*pPacket)); if (!pPacket) { WARNING_OUT(("CA2xSendMsg: can't get packet to send")); WARNING_OUT((" msg 0x%08x", msg)); WARNING_OUT((" data1 0x%08x", data1)); WARNING_OUT((" data2 0x%08x", data2));
DC_QUIT; }
pPacket->header.data.dataType = DT_CA; pPacket->msg = (TSHR_UINT16)msg; pPacket->data1 = (TSHR_UINT16)data1; pPacket->data2 = data2;
#ifdef _DEBUG
sentSize = #endif
DCS_CompressAndSendPacket(PROT_STR_MISC, g_s20BroadcastID, &(pPacket->header), sizeof(*pPacket)); TRACE_OUT(("CA request packet size: %08d, sent %08d", sizeof(*pPacket), sentSize));
fSent = TRUE;
DC_EXIT_POINT:
DebugExitBOOL(ASShare::CA2xSendMsg, fSent); return(fSent); }
//
// CA2xQueueSendMsg()
// This sends (or queues if failure) a 2.x node CA message. It has different
// fields, hence a different routine.
//
BOOL ASShare::CA2xQueueSendMsg ( UINT_PTR destID, UINT msg, UINT_PTR data1, UINT_PTR data2 ) { BOOL rc = TRUE; PCAREQUEST pCARequest;
DebugEntry(ASShare::CA2xQueueSendMsg);
if (msg != CA_OLDMSG_GRANTED_CONTROL) { ASSERT(!destID); }
//
// A DETACH message will cancel out a pending GRANTED_CONTROL message.
// So look for that first. If we find one (and there can only be at
// most one), replace it.
//
if (msg == CA_OLDMSG_DETACH) { pCARequest = (PCAREQUEST)COM_BasedListFirst(&m_caQueuedMsgs, FIELD_OFFSET(CAREQUEST, chain)); while (pCARequest) { if ((pCARequest->type == REQUEST_2X) && (pCARequest->destID == destID) && (pCARequest->msg == CA_OLDMSG_GRANTED_CONTROL)) { // Replace it
WARNING_OUT(("Replacing cancelled GRANTED_CONTROL msg to 2.x host"));
pCARequest->destID = 0; pCARequest->msg = CA_OLDMSG_DETACH; pCARequest->req.req2x.data1 = 0; pCARequest->req.req2x.data2 = 0;
// We're done.
DC_QUIT; }
pCARequest = (PCAREQUEST)COM_BasedListNext(&m_caQueuedMsgs, pCARequest, FIELD_OFFSET(CAREQUEST, chain)); } } //
// The messages must go out in order. So we must flush pending
// queued messages first.
//
if (!CAFlushOutgoingPackets() || !CA2xSendMsg(destID, msg, data1, data2)) { //
// We must queue this.
//
WARNING_OUT(("CA2xQueueSendMsg: queueing request for send later"));
pCARequest = new CAREQUEST; if (!pCARequest) { ERROR_OUT(("CA2xQueueSendMsg: can't even allocate memory to queue request; must fail")); rc = FALSE; } else { SET_STAMP(pCARequest, CAREQUEST);
pCARequest->type = REQUEST_2X; pCARequest->destID = destID; pCARequest->msg = msg; pCARequest->req.req2x.data1 = data1; pCARequest->req.req2x.data2 = data2;
//
// Stick this at the end of the queue
//
COM_BasedListInsertBefore(&(m_caQueuedMsgs), &(pCARequest->chain)); } }
DC_EXIT_POINT: DebugExitBOOL(ASShare::CA2xQueueSendMsg, rc); return(rc); }
//
// CAStartQuery()
//
// This puts up the modeless dialog to query the user about a control
// request. It will timeout if not handled.
//
BOOL ASShare::CAStartQuery ( ASPerson * pasFrom, UINT msg, PCA30P pReq ) { BOOL rc = FALSE;
DebugEntry(ASShare::CAStartQuery);
ValidatePerson(pasFrom);
//
// We have no stacked queries. If another comes in while the current
// one is up, it gets an immediate failure busy.
//
ASSERT(!m_caQueryDlg); ASSERT(!m_caQuery.pasReplyTo); ASSERT(!m_caQuery.msg);
//
// Setup for new query
//
if (msg == CA_PREFER_PASSCONTROL) { //
// With forwarding, the person we're going to send a packet to
// if accepted is not the person who sent us the request. It's the
// person we're forwarding to.
//
m_caQuery.pasReplyTo = SC_PersonFromNetID(pReq->ppc.mcsPassTo); ValidatePerson(m_caQuery.pasReplyTo); } else { m_caQuery.pasReplyTo = pasFrom; } m_caQuery.mcsOrg = pasFrom->mcsID; m_caQuery.msg = msg; m_caQuery.request = *pReq;
//
// If we are unattended, or the requester is unattended, instantly
// confirm. That's why we show the window after creating the dialog.
//
if ((m_pasLocal->cpcCaps.general.typeFlags & AS_UNATTENDED) || (pasFrom->cpcCaps.general.typeFlags & AS_UNATTENDED)) { CAFinishQuery(CARESULT_CONFIRMED); rc = TRUE; } else { //
// If this is a request to us && we're hosting, check auto-accept/
// auto-reject settings.
//
if (m_pHost && ((msg == CA_REQUEST_TAKECONTROL) || (msg == CA_PREFER_PASSCONTROL))) { if (m_pHost->m_caTempRejectRequests) { CAFinishQuery(CARESULT_DENIED_BUSY); rc = TRUE; DC_QUIT; } else if (m_pHost->m_caAutoAcceptRequests) { CAFinishQuery(CARESULT_CONFIRMED); rc = TRUE; DC_QUIT; } }
m_caQueryDlg = CreateDialogParam(g_asInstance, MAKEINTRESOURCE(IDD_QUERY), NULL, CAQueryDlgProc, 0); if (!m_caQueryDlg) { ERROR_OUT(("Failed to create query message box from [%d]", pasFrom->mcsID));
m_caQuery.pasReplyTo = NULL; m_caQuery.mcsOrg = 0; m_caQuery.msg = 0; } else { // Success
rc = TRUE; } }
DC_EXIT_POINT: DebugExitBOOL(ASShare::CAStartQuery, rc); return(rc); }
//
// CAFinishQuery()
//
// Called to finish the query we started, either because of UI or because
// we or the remote are unattended.
//
void ASShare::CAFinishQuery(UINT result) { CA30PENDING request;
DebugEntry(ASShare::CAFinishQuery);
ValidatePerson(m_caQuery.pasReplyTo);
// Make a copy of our request
request = m_caQuery;
//
// If we have a dialog up, destroy it NOW. Completing the request
// may cause us to be controlled or whatever. So get the dialog
// out of the way immediately.
//
// Note that destroying ourself will clear the request vars, hence the
// copy above.
//
if (m_caQueryDlg) { DestroyWindow(m_caQueryDlg); } else { m_caQuery.pasReplyTo = NULL; m_caQuery.mcsOrg = 0; m_caQuery.msg = 0; }
switch (request.msg) { case CA_REQUEST_TAKECONTROL: { CACompleteRequestTakeControl(request.pasReplyTo, &request.request.rtc, result); break; }
case CA_REQUEST_GIVECONTROL: { CACompleteRequestGiveControl(request.pasReplyTo, &request.request.rgc, result); break; }
case CA_PREFER_PASSCONTROL: { CACompletePreferPassControl(request.pasReplyTo, request.mcsOrg, &request.request.ppc, result); break; }
default: { ERROR_OUT(("Unrecognized query msg %d", request.msg)); break; } }
DebugExitVOID(ASShare::CAFinishQuery); }
//
// CA_QueryDlgProc()
//
// Handles querying user dialog
//
INT_PTR CALLBACK CAQueryDlgProc ( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam ) { return(g_asSession.pShare->CA_QueryDlgProc(hwnd, message, wParam, lParam)); }
BOOL ASShare::CA_QueryDlgProc ( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam ) { BOOL rc = TRUE;
DebugEntry(CA_QueryDlgProc);
switch (message) { case WM_INITDIALOG: { char szT[256]; char szRes[512]; char szShared[64]; UINT idsTitle; ASPerson * pasT; HDC hdc; HFONT hfn; RECT rc; RECT rcOwner;
ValidatePerson(m_caQuery.pasReplyTo);
pasT = NULL;
// Set title.
ASSERT(m_caQuery.msg); switch (m_caQuery.msg) { case CA_REQUEST_TAKECONTROL: { idsTitle = IDS_TITLE_QUERY_TAKECONTROL;
if (m_pasLocal->hetCount == HET_DESKTOPSHARED) LoadString(g_asInstance, IDS_DESKTOP_LOWER, szShared, sizeof(szShared)); else LoadString(g_asInstance, IDS_PROGRAMS_LOWER, szShared, sizeof(szShared));
LoadString(g_asInstance, IDS_MSG_QUERY_TAKECONTROL, szT, sizeof(szT));
wsprintf(szRes, szT, m_caQuery.pasReplyTo->scName, szShared); break; }
case CA_REQUEST_GIVECONTROL: { if (m_caQuery.pasReplyTo->hetCount == HET_DESKTOPSHARED) LoadString(g_asInstance, IDS_DESKTOP_LOWER, szShared, sizeof(szShared)); else LoadString(g_asInstance, IDS_PROGRAMS_LOWER, szShared, sizeof(szShared));
if (m_caQuery.request.rgc.mcsPassFrom) { pasT = SC_PersonFromNetID(m_caQuery.request.rgc.mcsPassFrom); }
if (pasT) { idsTitle = IDS_TITLE_QUERY_YIELDCONTROL;
LoadString(g_asInstance, IDS_MSG_QUERY_YIELDCONTROL, szT, sizeof(szT));
wsprintf(szRes, szT, pasT->scName, m_caQuery.pasReplyTo->scName, szShared); } else { idsTitle = IDS_TITLE_QUERY_GIVECONTROL;
LoadString(g_asInstance, IDS_MSG_QUERY_GIVECONTROL, szT, sizeof(szT));
wsprintf(szRes, szT, m_caQuery.pasReplyTo->scName, szShared); }
break; }
case CA_PREFER_PASSCONTROL: { pasT = SC_PersonFromNetID(m_caQuery.mcsOrg); ValidatePerson(pasT);
idsTitle = IDS_TITLE_QUERY_FORWARDCONTROL;
if (m_pasLocal->hetCount == HET_DESKTOPSHARED) LoadString(g_asInstance, IDS_DESKTOP_LOWER, szShared, sizeof(szShared)); else LoadString(g_asInstance, IDS_PROGRAMS_LOWER, szShared, sizeof(szShared));
LoadString(g_asInstance, IDS_MSG_QUERY_FORWARDCONTROL, szT, sizeof(szT));
wsprintf(szRes, szT, pasT->scName, szShared, m_caQuery.pasReplyTo->scName);
break; }
default: { ERROR_OUT(("Bogus m_caQuery.msg %d", m_caQuery.msg)); break; } }
LoadString(g_asInstance, idsTitle, szT, sizeof(szT)); SetWindowText(hwnd, szT);
// Set message.
SetDlgItemText(hwnd, CTRL_QUERY, szRes);
// Center the message vertically
GetWindowRect(GetDlgItem(hwnd, CTRL_QUERY), &rcOwner); MapWindowPoints(NULL, hwnd, (LPPOINT)&rcOwner, 2);
rc = rcOwner;
hdc = GetDC(hwnd); hfn = (HFONT)SendDlgItemMessage(hwnd, CTRL_QUERY, WM_GETFONT, 0, 0); hfn = SelectFont(hdc, hfn);
DrawText(hdc, szRes, -1, &rc, DT_NOCLIP | DT_EXPANDTABS | DT_NOPREFIX | DT_WORDBREAK | DT_CALCRECT);
SelectFont(hdc, hfn); ReleaseDC(hwnd, hdc);
ASSERT((rc.bottom - rc.top) <= (rcOwner.bottom - rcOwner.top));
SetWindowPos(GetDlgItem(hwnd, CTRL_QUERY), NULL, rcOwner.left, ((rcOwner.top + rcOwner.bottom) - (rc.bottom - rc.top)) / 2, (rcOwner.right - rcOwner.left), rc.bottom - rc.top, SWP_NOACTIVATE | SWP_NOZORDER);
SetTimer(hwnd, IDT_CAQUERY, PERIOD_CAQUERY, 0);
//
// Show window, the user will handle
//
ShowWindow(hwnd, SW_SHOWNORMAL); SetForegroundWindow(hwnd); UpdateWindow(hwnd);
break; }
case WM_COMMAND: { switch (GET_WM_COMMAND_ID(wParam, lParam)) { case IDOK: { CAFinishQuery(CARESULT_CONFIRMED); break; }
case IDCANCEL: { CAFinishQuery(CARESULT_DENIED_USER); break; } } break; }
case WM_TIMER: { if (wParam != IDT_CAQUERY) { rc = FALSE; } else { KillTimer(hwnd, IDT_CAQUERY);
// Timed out failure.
CAFinishQuery(CARESULT_DENIED_TIMEDOUT); } break; }
case WM_DESTROY: { //
// Clear pending info
//
m_caQueryDlg = NULL; m_caQuery.pasReplyTo = NULL; m_caQuery.mcsOrg = 0; m_caQuery.msg = 0; break; }
default: { rc = FALSE; break; } }
DebugExitBOOL(CA_QueryDlgProc, rc); return(rc); }
//
// CACancelQuery()
//
// If a dialog is up for a take control request, it hasn't been handled yet,
// and we get a cancel notification from the viewer, we need to take the
// dialog down WITHOUT generating a response packet.
//
void ASShare::CACancelQuery ( ASPerson * pasFrom, BOOL fPacket ) { DebugEntry(ASShare::CACancelQuery);
ASSERT(m_caQueryDlg); ASSERT(m_caQuery.pasReplyTo == pasFrom);
if (fPacket) { // This will send a packet then destroy the dialog
CAFinishQuery(CARESULT_DENIED); } else { // Destroy the dialog
DestroyWindow(m_caQueryDlg); }
ASSERT(!m_caQueryDlg); ASSERT(!m_caQuery.pasReplyTo); ASSERT(!m_caQuery.msg);
DebugExitVOID(ASShare::CACancelQuery); }
|