1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675 |
- /* OpenDoors Online Software Programming Toolkit
- * (C) Copyright 1991 - 1999 by Brian Pirie.
- *
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public
- * License as published by the Free Software Foundation; either
- * version 2 of the License, or (at your option) any later version.
- *
- * This library 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
- * Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public
- * License along with this library; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
- *
- *
- * File: ODKrnl.c
- *
- * Description: Contains the OpenDoors kernel, which is responsible for many
- * of the core functions which continue regardless of what the
- * client program is doing. The implementation of this file is
- * central to the OpenDoors architecture. The functionality
- * implemented by the OpenDoors kernel includes (but is not
- * limited to):
- *
- * - Obtaining and input from the user, through the modem
- * and possibly the local keyboard.
- * - Monitoring maximum time and inactivity time limits.
- * - Responding to loss of carrier.
- * - Forcing the status line to be updated regularily,
- * on platforms that it exists.
- * - Implementing the system operator <-> remote user chat
- * mode.
- *
- * Revisions: Date Ver Who Change
- * ---------------------------------------------------------------
- * Jan 01, 1995 6.00 BP Split off from odcore.c
- * Nov 11, 1995 6.00 BP Removed register keyword.
- * Nov 14, 1995 6.00 BP Added include of odscrn.h.
- * Nov 15, 1995 6.00 BP 32-bit portability.
- * Nov 16, 1995 6.00 BP Removed oddoor.h, added odcore.h.
- * Nov 17, 1995 6.00 BP Use new input queue mechanism.
- * Nov 21, 1995 6.00 BP Ported to Win32.
- * Dec 12, 1995 6.00 BP Added entry, exit and kernel macros.
- * Dec 13, 1995 6.00 BP Moved chat mode code to ODKrnl.h.
- * Dec 24, 1995 6.00 BP od_chat_active = TRUE on chat start.
- * Dec 30, 1995 6.00 BP Added ODCALL for calling convention.
- * Jan 04, 1996 6.00 BP tODInQueueEvent -> tODInputEvent.
- * Jan 12, 1996 6.00 BP Added bOnlyShiftArrow.
- * Jan 30, 1996 6.00 BP Replaced od_yield() with od_sleep().
- * Jan 30, 1996 6.00 BP Add semaphore timeout.
- * Feb 06, 1996 6.00 BP Added od_silent_mode.
- * Feb 19, 1996 6.00 BP Changed version number to 6.00.
- * Feb 23, 1996 6.00 BP Only create active semapore once.
- * Mar 03, 1996 6.10 BP Begin version 6.10.
- * Mar 06, 1996 6.10 BP Prevent TC generated N_SCOPY@ call.
- * Mar 13, 1996 6.10 BP bOnlyShiftArrow -> nArrowUseCount.
- * Mar 19, 1996 6.10 BP MSVC15 source-level compatibility.
- * Oct 22, 2001 6.21 RS Lowered thread priorities to normal.
- * Aug 10, 2003 6.23 SH *nix support
- */
- #define BUILDING_OPENDOORS
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <ctype.h>
- #include <time.h>
- #include <limits.h>
- #include "OpenDoor.h"
- #ifdef ODPLAT_NIX
- #include <sys/types.h>
- #include <unistd.h>
- #include <signal.h>
- #include <fcntl.h>
- #include <errno.h>
- #endif
- #include "ODCore.h"
- #include "ODGen.h"
- #include "ODPlat.h"
- #include "ODCom.h"
- #include "ODKrnl.h"
- #include "ODScrn.h"
- #include "ODInQue.h"
- #include "ODInEx.h"
- #ifdef ODPLAT_WIN32
- #include "ODFrame.h"
- #endif /* ODPLAT_WIN32 */
- /* Multithreading performance tuning. */
- #define REMOTE_INPUT_THREAD_PRIORITY OD_PRIORITY_NORMAL /* was ABOVE_NORMAL */
- #define NO_CARRIER_THREAD_PRIORITY OD_PRIORITY_NORMAL /* was ABOVE_NORMAL */
- #define NO_CARRIER_THREAD_SLEEP_TIME 6000
- #define TIME_UPDATE_THREAD_PRIORITY OD_PRIORITY_NORMAL
- #define TIME_UPDATE_THREAD_SLEEP_TIME 3000
- /* Misc performance tuning. */
- #define STATUS_UPDATE_PERIOD 3L
- #define CHAT_YIELD_PERIOD 25L
- /* Pending command identifiers. */
- #define KERNEL_FUNC_CHATTOGGLE 0x0001
- /* Private function prototypes. */
- static void ODKrnlHandleReceivedChar(char chReceived, BOOL bFromRemote);
- static void ODKrnlTimeUpdate(void);
- static void ODKrnlChatCleanup(void);
- static void ODKrnlChatMode(void);
- #ifdef ODPLAT_NIX
- #ifdef USE_KERNEL_SIGNAL
- static void sig_run_kernel(int sig);
- static void sig_get_char(int sig);
- static void sig_no_carrier(int sig);
- #endif
- #endif
- /* Functions specific to the multithreaded implementation of the kernel. */
- #ifdef OD_MULTITHREADED
- /* Thread proceedures. */
- DWORD OD_THREAD_FUNC ODKrnlRemoteInputThread(void *pParam);
- DWORD OD_THREAD_FUNC ODKrnlNoCarrierThread(void *pParam);
- DWORD OD_THREAD_FUNC ODKrnlTimeUpdateThread(void *pParam);
- DWORD OD_THREAD_FUNC ODKrnlChatThread(void *pParam);
- /* Helper functions. */
- static void ODKrnlWaitForExclusiveControl(void);
- static void ODKrnlGiveUpExclusiveControl(void);
- #endif /* OD_MULTITHREADED */
- /* Local working variables. */
- #ifdef OD_MULTITHREADED
- static tODThreadHandle hRemoteInputThread = NULL;
- static tODThreadHandle hNoCarrierThread = NULL;
- static tODThreadHandle hTimeUpdateThread = NULL;
- static tODThreadHandle hClientThread = NULL;
- static tODThreadHandle hChatThread = NULL;
- static BOOL bHaveExclusiveControl;
- static BOOL bChatActivatedInternally;
- #endif /* OD_MULTITHREADED */
- static BOOL bKernelActive = FALSE;
- static BOOL bWarnedAboutInactivity = FALSE;
- static INT16 nLastInactivitySetting = 0;
- static time_t nNextStatusUpdateTime;
- static INT nKrnlFuncPending;
- static BOOL bLastStatusSetting;
- static INT16 nChatOriginalAttrib;
- /* Global kernel-related variables. */
- tODTimer RunKernelTimer;
- time_t nNextTimeDeductTime;
- char chLastControlKey = '\0';
- INT nArrowUseCount = 0;
- BOOL bForceStatusUpdate = FALSE;
- BOOL bIsShell;
- #ifdef OD_MULTITHREADED
- tODSemaphoreHandle hODActiveSemaphore = NULL;
- #endif /* OD_MULTITHREADED */
- /* ========================================================================= */
- /* Core of the OpenDoors Kernel. */
- /* ========================================================================= */
- /* ----------------------------------------------------------------------------
- * ODKrnlInitialize()
- *
- * Initializes kernel activities. In multithreaded versions of OpenDoors, this
- * is the function that starts the various kernel threads.
- *
- * Parameters: kODRCSuccess on success, or an error code on failure.
- *
- * Return: void
- */
- tODResult ODKrnlInitialize(void)
- {
- #ifdef ODPLAT_NIX
- sigset_t block;
- #ifdef USE_KERNEL_SIGNAL
- struct sigaction act;
- struct itimerval itv;
- #endif
- #endif
- tODResult Result = kODRCSuccess;
-
- #ifdef ODPLAT_NIX
- #ifdef USE_KERNEL_SIGNAL
- /* HUP Detection */
- act.sa_handler=sig_no_carrier;
- /* If two HUP signals are recieved, die on the second */
- act.sa_flags=SA_RESETHAND|SA_RESTART;
- sigemptyset(&(act.sa_mask));
- sigaction(SIGHUP,&act,NULL);
- /* Run kernel on SIGALRM (Every .01 seconds) */
- act.sa_handler=sig_run_kernel;
- act.sa_flags=SA_RESTART;
- sigemptyset(&(act.sa_mask));
- sigaction(SIGALRM,&act,NULL);
- itv.it_interval.tv_sec=0;
- itv.it_interval.tv_usec=10000;
- itv.it_value.tv_sec=0;
- itv.it_value.tv_usec=10000;
- setitimer(ITIMER_REAL,&itv,NULL);
- /* Make stdin signal driven. */
- // act.sa_handler=sig_get_char;
- // act.sa_flags=0;
- // sigemptyset(&(act.sa_mask));
- // sigaction(SIGIO,&act,NULL);
- //
- // /* Have SIGIO signals delivered to this process */
- // fcntl(0,F_SETOWN,getpid());
- //
- // /* Enable SIGIO when read possible on stdin */
- // fcntl(0,F_SETFL,fcntl(0,F_GETFL)|O_ASYNC);
- /* Make sure SIGHUP, SIGALRM, and SIGIO are unblocked */
- sigemptyset(&block);
- sigaddset(&block,SIGHUP);
- sigaddset(&block,SIGALRM);
- #if 0
- sigaddset(&block,SIGIO);
- #endif
- sigprocmask(SIG_UNBLOCK,&block,NULL);
- #else /* Using ODComCarrier... don't catch HUP signal */
- sigemptyset(&block);
- sigaddset(&block,SIGHUP);
- sigprocmask(SIG_BLOCK,&block,NULL);
- #endif
- #endif
- /* Initialize time of next status update and next time deduction. */
- nNextStatusUpdateTime = time(NULL) + STATUS_UPDATE_PERIOD;
- nNextTimeDeductTime = time(NULL) + 60L;
- bLastStatusSetting = od_control.od_status_on = TRUE;
- /* Initially, no kernel functions are pending. */
- nKrnlFuncPending = 0;
- /* Initially, the kernel is not active. */
- bKernelActive = FALSE;
- #ifdef OD_MULTITHREADED
- /* Initially, we do not have exclusive control of the application. */
- bHaveExclusiveControl = FALSE;
- /* Obtain a handle to the client thread. */
- hClientThread = ODThreadGetCurrent();
- /* Create OpenDoors activation semaphore. */
- if(hODActiveSemaphore == NULL)
- {
- Result = ODSemaphoreAlloc(&hODActiveSemaphore, 0, INT_MAX);
- if(Result != kODRCSuccess) return(Result);
- }
- /* Start the remote input thread if we are not operating in local mode. */
- if(od_control.baud != 0)
- {
- Result = ODThreadCreate(&hRemoteInputThread, ODKrnlRemoteInputThread,
- NULL);
- if(Result != kODRCSuccess) return(Result);
- ODThreadSetPriority(hRemoteInputThread, REMOTE_INPUT_THREAD_PRIORITY);
- }
- /* Start the carrier detection thread if we are not operating in local */
- /* mode. */
- if(od_control.baud != 0)
- {
- Result = ODThreadCreate(&hNoCarrierThread, ODKrnlNoCarrierThread, NULL);
- if(Result != kODRCSuccess) return(Result);
- ODThreadSetPriority(hNoCarrierThread, NO_CARRIER_THREAD_PRIORITY);
- }
- /* Start the time update thread. */
- Result = ODThreadCreate(&hTimeUpdateThread, ODKrnlTimeUpdateThread, 0);
- if(Result != kODRCSuccess) return(Result);
- ODThreadSetPriority(hTimeUpdateThread, TIME_UPDATE_THREAD_PRIORITY);
- #endif /* OD_MULTITHREADED */
- /* Return with success. */
- return(Result);
- }
- /* ----------------------------------------------------------------------------
- * ODKrnlShutdown()
- *
- * Shuts down kernel activities.
- *
- * Parameters: none
- *
- * Return: void
- */
- void ODKrnlShutdown(void)
- {
- if(bKernelActive) return;
- #ifdef OD_MULTITHREADED
- #if defined(OD_DIAGNOSTICS) && defined(ODPLAT_WIN32)
- if(od_control.od_internal_debug)
- MessageBox(NULL, "Terminating remote input thread", "OpenDoors Diagnostics", MB_OK);
- #endif
- /* Shutdown the remote input thread, if it exists. */
- if(hRemoteInputThread != NULL) ODThreadTerminate(hRemoteInputThread);
- #if defined(OD_DIAGNOSTICS) && defined(ODPLAT_WIN32)
- if(od_control.od_internal_debug)
- MessageBox(NULL, "Terminating carrier detection", "OpenDoors Diagnostics", MB_OK);
- #endif
- /* Shutdown the carrier detection thread, if it exists. */
- if(hNoCarrierThread != NULL) ODThreadTerminate(hNoCarrierThread);
- #if defined(OD_DIAGNOSTICS) && defined(ODPLAT_WIN32)
- if(od_control.od_internal_debug)
- MessageBox(NULL, "Terminating time update thread", "OpenDoors Diagnostics", MB_OK);
- #endif
- /* Shutdown the time update thread, if it exists. */
- if(hTimeUpdateThread != NULL) ODThreadTerminate(hTimeUpdateThread);
- #if defined(OD_DIAGNOSTICS) && defined(ODPLAT_WIN32)
- if(od_control.od_internal_debug)
- MessageBox(NULL, "Releasing activation semaphore", "OpenDoors Diagnostics", MB_OK);
- #endif
- #endif /* OD_MULTITHREADED */
- }
- /* ----------------------------------------------------------------------------
- * od_kernel()
- *
- * Carries out any kernel tasks that must be performed through regular,
- * explicit calls to this function,
- *
- * Parameters: none
- *
- * Return: void
- */
- ODAPIDEF void ODCALL od_kernel(void)
- {
- #ifndef OD_MULTITHREADED
- char ch;
- #ifdef ODPLAT_DOS
- WORD wKey;
- BYTE btShiftStatus;
- char *pszShellName;
- #endif
- BOOL bCarrier;
- #endif /* OD_MULTITHREADED */
- /* Log function entry if running in trace mode. */
- TRACE(TRACE_API, "od_kernel()");
- /* Initialize OpenDoors if not already done. */
- if(!bODInitialized) od_init();
- /* If this is an attempt at a re-entrant call to od_kernel() from another */
- /* function called by a currently active od_kernel(), then return without */
- /* doing anything. */
- if(bKernelActive) return;
- OD_API_ENTRY();
- /* Note that kernel is active to prevent recursive calls to the kernel. */
- bKernelActive = TRUE;
- /* Call od_ker_exec function if required. */
- if(od_control.od_ker_exec != NULL)
- {
- (*od_control.od_ker_exec)();
- }
- /* The remainder of od_kernel() only applies to non-multithreaded */
- /* versions of OpenDoors. */
- #ifndef OD_MULTITHREADED
- /* If not operating in local mode, then perform remote-mode specific */
- /* activies. */
- if(od_control.baud != 0)
- {
- #ifndef USE_KERNEL_SIGNAL
- /* If carrier detection is enabled, then shutdown OpenDoors if */
- /* the carrier detect signal is no longer high. */
- if(!(od_control.od_disable&DIS_CARRIERDETECT))
- {
- ODComCarrier(hSerialPort, &bCarrier);
- if(!bCarrier)
- {
- ODKrnlForceOpenDoorsShutdown(ERRORLEVEL_NOCARRIER);
- }
- }
- #endif
- /* Loop, obtaining any new characters from the serial port and */
- /* adding them to the common local/remote input queue. */
- while(ODComGetByte(hSerialPort, &ch, FALSE) == kODRCSuccess)
- {
- ODKrnlHandleReceivedChar(ch, TRUE);
- }
- }
- #ifdef ODPLAT_DOS
- check_keyboard_again:
- if(nKrnlFuncPending && !bShellChatActive)
- {
- if(nKrnlFuncPending & KERNEL_FUNC_CHATTOGGLE)
- {
- nKrnlFuncPending &=~ KERNEL_FUNC_CHATTOGGLE;
- goto chat_pressed;
- }
- }
- /* Don't check local keyboard if sysop DIS_SYSOP_KEYS is set, or if we */
- /* are operatingin silent mode. */
- if(od_control.od_disable & DIS_SYSOP_KEYS
- || od_control.od_silent_mode)
- {
- goto after_key_check;
- }
- ASM mov ah, 1
- ASM push si
- ASM push di
- ASM int 0x16
- ASM jnz key_waiting
- ASM pop di
- ASM pop si
- ASM jmp after_key_check
- key_waiting:
- ASM mov ah, 0
- ASM int 0x16
- ASM mov wKey, ax
- ASM mov ah, 2
- ASM int 0x16
- ASM mov btShiftStatus, al
- ASM pop di
- ASM pop si
- if(nArrowUseCount > 0 && (wKey == 0x4800 || wKey == 0x5000)
- && !(btShiftStatus & 2))
- {
- /* Pass key on to od_local_input, if it is defined. */
- if(od_control.od_local_input != NULL)
- {
- (*od_control.od_local_input)(wKey);
- }
- /* Add this key to the local/remote input queue. */
- ODKrnlHandleLocalKey(wKey);
- }
- /* If hangup key is pressed. */
- else if(wKey == od_control.key_hangup)
- {
- ODKrnlForceOpenDoorsShutdown(ERRORLEVEL_HANGUP);
- }
- /* If drop to BBS key is pressed. */
- else if(wKey == od_control.key_drop2bbs)
- {
- ODKrnlForceOpenDoorsShutdown(ERRORLEVEL_DROPTOBBS);
- }
- else if(wKey == od_control.key_dosshell)
- {
- if(!bShellChatActive)
- {
- if(pfLogWrite != NULL)
- (*pfLogWrite)(6);
- /* If function hook is defined. */
- if(od_control.od_cbefore_shell != NULL)
- {
- /* Then call it. */
- bShellChatActive = TRUE;
- (*od_control.od_cbefore_shell)();
- bShellChatActive = FALSE;
- }
- if(od_control.od_before_shell != NULL)
- od_disp_str(od_control.od_before_shell);
- if((pszShellName = (char *)getenv("COMSPEC")) == NULL)
- {
- pszShellName = (char *)"COMMAND.COM";
- }
- bIsShell = TRUE;
- od_spawnvpe(P_WAIT, pszShellName, NULL, NULL);
- bIsShell = FALSE;
- if(od_control.od_after_shell != NULL)
- od_disp_str(od_control.od_after_shell);
- /* If a function hook is defined. */
- if(od_control.od_cafter_shell != NULL)
- {
- /* Then call it. */
- bShellChatActive = TRUE;
- (*od_control.od_cafter_shell)();
- bShellChatActive = FALSE;
- }
- if(pfLogWrite != NULL)
- (*pfLogWrite)(7);
- }
- }
- /* If toggle chat mode key is pressed. */
- else if(wKey == od_control.key_chat)
- {
- chat_pressed:
- if(!bShellChatActive || od_control.od_chat_active)
- {
- /* If chat mode is active. */
- if(od_control.od_chat_active)
- {
- /* Signal exit of chat mode. */
- ODKrnlEndChatMode();
- }
- /* If chat mode is off. */
- else
- {
- /* Enable second call to kernel. */
- bKernelActive = FALSE;
- /* Enter chat mode. */
- ODKrnlChatMode();
- /* Disable second call to kernel. */
- bKernelActive = TRUE;
- }
- }
- else
- {
- if(nKrnlFuncPending & KERNEL_FUNC_CHATTOGGLE)
- {
- nKrnlFuncPending &= ~KERNEL_FUNC_CHATTOGGLE;
- }
- else
- {
- nKrnlFuncPending |= KERNEL_FUNC_CHATTOGGLE;
- }
- }
- }
- /* If sysop next key is pressed. */
- else if(wKey == od_control.key_sysopnext)
- {
- /* Toggle sysop next setting. */
- od_control.sysop_next = !od_control.sysop_next;
- /* Update status line. */
- goto statup;
- }
- /* If ESCape key is pressed and we are in chat mode. */
- else if((wKey&0xff) == 27 && od_control.od_chat_active)
- {
- /* Signal exit from chat mode. */
- od_control.od_chat_active = FALSE;
- }
- /* If lockout user key is pressed. */
- else if(wKey == od_control.key_lockout)
- {
- /* Set the user's access security level to 0. */
- od_control.user_security = 0;
- /* Shutdown OpenDoors. */
- ODKrnlForceOpenDoorsShutdown(ERRORLEVEL_HANGUP);
- }
- /* If toggle keyboard off key is pressed. */
- else if(wKey == od_control.key_keyboardoff)
- {
- /* Toggle user keyboard settings. */
- od_control.od_user_keyboard_on =! od_control.od_user_keyboard_on;
- /* Update status line. */
- goto statup;
- }
- /* If increase time key is pressed. */
- else if(wKey == od_control.key_moretime)
- {
- /* If time limit is less than maximum possible time limit. */
- if(od_control.user_timelimit < 1440)
- {
- /* Increase time left online. */
- ++od_control.user_timelimit;
- }
- /* Update status line. */
- goto statup;
- }
- /* If decrease time key is pressed. */
- else if(wKey == od_control.key_lesstime)
- {
- /* Never let user's time limit be set to a negative value. */
- if(od_control.user_timelimit > 0)
- {
- /* Decrease user's timelimit. */
- --od_control.user_timelimit;
- }
- /* Update the status line. */
- goto statup;
- }
- else
- {
- for(ch = 0; ch < 9; ++ch)
- {
- if(wKey == od_control.key_status[ch])
- {
- if(btCurrentStatusLine != ch && od_control.od_status_on)
- {
- od_set_statusline(ch);
- }
- goto check_keyboard_again;
- }
- }
- /* Look for user-defined hotkeys. */
- for(ch=0; ch<od_control.od_num_keys; ++ch)
- {
- /* If it matches. */
- if(wKey == (WORD)od_control.od_hot_key[ch])
- {
- /* Record keypress. */
- od_control.od_last_hot = wKey;
- /* Notify the current personality. */
- (*pfCurrentPersonality)(21);
- /* Check for a hotkey function. */
- if(od_control.od_hot_function[ch] != NULL)
- {
- /* Call it if it exists. */
- (*od_control.od_hot_function[ch])();
- }
- /* Stop searching. */
- break;
- }
- }
- /* If no hotkeys found. */
- if(ch >= od_control.od_num_keys)
- {
- /* Pass key on to od_local_input, if it is defined. */
- if(od_control.od_local_input != NULL)
- {
- (*od_control.od_local_input)(wKey);
- }
- /* Add this key to the local/remote input queue. */
- ODKrnlHandleLocalKey(wKey);
- }
- }
- goto check_keyboard_again;
- after_key_check:
- /* If status line has been turned on since last call to kernel. */
- if(bLastStatusSetting != od_control.od_status_on)
- {
- /* Generate the status line. */
- od_set_statusline(0);
- }
- bLastStatusSetting = od_control.od_status_on;
- if(od_control.od_update_status_now)
- {
- od_set_statusline(btCurrentStatusLine);
- od_control.od_update_status_now = FALSE;
- }
- /* Update status line when needed. */
- if(nNextStatusUpdateTime < time(NULL) || bForceStatusUpdate)
- {
- statup:
- nNextStatusUpdateTime = time(NULL) + STATUS_UPDATE_PERIOD;
- /* Turn off status line update force flag */
- bForceStatusUpdate = FALSE;
- if(od_control.od_status_on && btCurrentStatusLine != 8)
- {
- /* Store console settings. */
- ODStoreTextInfo();
- /* Enable writes to whole screen. */
- ODScrnSetBoundary(1, 1, 80, 25);
- ODScrnEnableCaret(FALSE);
- (*pfCurrentPersonality)((BYTE)(10 + btCurrentStatusLine));
- ODRestoreTextInfo();
- ODScrnEnableCaret(TRUE);
- }
- }
- #endif
- ODKrnlTimeUpdate();
- ODTimerStart(&RunKernelTimer, 250);
- OD_API_EXIT();
- bKernelActive = FALSE;
- #endif /* !OD_MULTITHREADED */
- }
- /* ----------------------------------------------------------------------------
- * ODKrnlHandleLocalKey()
- *
- * Called when a key is pressed on the local keyboard that should be placed
- * in the common local/remote input queue. This function is not called for
- * sysop function keys.
- *
- * Parameters: wKeyCode
- *
- * Return: void
- */
- void ODKrnlHandleLocalKey(WORD wKeyCode)
- {
- /* If local keyboard input by sysop has not been disabled. */
- if(!(od_control.od_disable & DIS_LOCAL_INPUT))
- {
- if((wKeyCode & 0xff) == 0)
- {
- ODKrnlHandleReceivedChar('\0', FALSE);
- ODKrnlHandleReceivedChar((char)(wKeyCode >> 8), FALSE);
- }
- else
- {
- ODKrnlHandleReceivedChar((char)wKeyCode, FALSE);
- }
- }
- }
- /* ----------------------------------------------------------------------------
- * ODKrnlHandleReceivedChar() *** PRIVATE FUNCTION ***
- *
- * Called when a character is received from the local or remote system.
- *
- * Parameters: chReceived - Character that should be handled.
- *
- * bFromRemote - TRUE if this character was received from the
- * remote system, FALSE if it originated from the
- * local console.
- *
- * Return: void
- */
- static void ODKrnlHandleReceivedChar(char chReceived, BOOL bFromRemote)
- {
- tODInputEvent InputEvent;
- /* If we are operating in remote mode, and remote user keyboard has been */
- /* disabled by the sysop, then return, ignoring this character. */
- if(bFromRemote && !od_control.od_user_keyboard_on)
- {
- return;
- }
- /* Add this input event to the local/remote common input queue. */
- InputEvent.EventType = EVENT_CHARACTER;
- InputEvent.bFromRemote = bFromRemote;
- InputEvent.chKeyPress = chReceived;
- ODInQueueAddEvent(hODInputQueue, &InputEvent);
- /* Update last control key information. */
- switch(chReceived)
- {
- case 's':
- case 'S':
- case 3:
- case 11:
- case 0x18:
- chLastControlKey = 's';
- break;
- case 'p':
- case 'P':
- chLastControlKey = 'p';
- }
- }
- /* ----------------------------------------------------------------------------
- * ODKrnlTimeUpdate() *** PRIVATE FUNCTION ***
- *
- * Performs regular updating of time remaining online, inactivity time, and
- * forces OpenDoors to exit if a time limit has been exceeded.
- *
- * Parameters: None
- *
- * Return: void
- */
- static void ODKrnlTimeUpdate(void)
- {
- time_t CurrentTime;
- static char szTemp[80];
- /* Obtain the current time. */
- CurrentTime = time(NULL);
- /* If inactivity setting has changed. */
- if(nLastInactivitySetting != od_control.od_inactivity)
- {
- /* If it was previously disabled. */
- if(nLastInactivitySetting == 0)
- {
- /* Prevent immediate timeout. */
- ODInQueueResetLastActivity(hODInputQueue);
- }
- /* Store current value. */
- nLastInactivitySetting = od_control.od_inactivity;
- }
- /* Check user keyboard inactivity. */
- if((ODInQueueGetLastActivity(hODInputQueue) + od_control.od_inactivity)
- < CurrentTime)
- {
- /* If timeout, display message. */
- if(od_control.od_inactivity != 0 && !od_control.od_disable_inactivity)
- {
- if(od_control.od_time_msg_func == NULL)
- {
- od_disp_str(od_control.od_inactivity_timeout);
- }
- else
- {
- (*od_control.od_time_msg_func)(od_control.od_inactivity_timeout);
- }
- /* End connection. */
- ODKrnlForceOpenDoorsShutdown(ERRORLEVEL_INACTIVITY);
- }
- }
- /* If less than 5s left of inactivity. */
- else if(ODInQueueGetLastActivity(hODInputQueue) + od_control.od_inactivity
- < CurrentTime + od_control.od_inactive_warning)
- {
- if(!bWarnedAboutInactivity && od_control.od_inactivity != 0
- && !od_control.od_disable_inactivity)
- {
- /* Warn the user. */
- if(od_control.od_time_msg_func == NULL)
- {
- od_disp_str(od_control.od_inactivity_warning);
- }
- else
- {
- (*od_control.od_time_msg_func)(od_control.od_inactivity_warning);
- }
- /* Don't warn the user a second time. */
- bWarnedAboutInactivity = TRUE;
- }
- }
- else
- {
- /* Re-enable inactivity warning. */
- bWarnedAboutInactivity = FALSE;
- }
- /* If chat mode is active. */
- if(od_control.od_chat_active)
- {
- /* Prevent the user's time from being drained. */
- nNextTimeDeductTime = time(NULL) + 60;
- }
- /* If 1 minute has passed since last time update. */
- if(CurrentTime >= nNextTimeDeductTime)
- {
- /* Next time update should occur 60 seconds after this one was */
- /* scheduled. */
- nNextTimeDeductTime += 60;
- /* Force status line to be updated immediately. */
- bForceStatusUpdate = TRUE;
- /* Decrement time left. */
- --od_control.user_timelimit;
- /* If the user's time limit is close to expiring, then notify */
- /* the user. */
- if(od_control.user_timelimit <= 3 &&
- od_control.user_timelimit > 0 &&
- !(od_control.od_disable & DIS_TIMEOUT))
- {
- /* If less than 3 mins left, tell user. */
- sprintf(szTemp, od_control.od_time_warning,
- od_control.user_timelimit);
- if(od_control.od_time_msg_func == NULL)
- {
- od_disp_str(szTemp);
- }
- else
- {
- (*od_control.od_time_msg_func)(szTemp);
- }
- }
- #ifdef ODPLAT_WIN32
- ODFrameUpdateTimeDisplay();
- #endif /* ODPLAT_WIN32 */
- }
- /* If user has no time left. */
- if(od_control.user_timelimit <= 0
- && !(od_control.od_disable & DIS_TIMEOUT))
- {
- /* Notify the user. */
- if(od_control.od_time_msg_func == NULL)
- {
- od_disp_str(od_control.od_no_time);
- }
- else
- {
- (*od_control.od_time_msg_func)(od_control.od_no_time);
- }
- /* Force OpenDoors to shutdown. */
- ODKrnlForceOpenDoorsShutdown(ERRORLEVEL_TIMEOUT);
- }
- }
- /* ----------------------------------------------------------------------------
- * ODKrnlForceOpenDoorsShutdown()
- *
- * Called to force the application to exit due to some event in OpenDoors,
- * such as loss of carrier, user inactivity timeout, the hangup command
- * being chosen by the system operator, etc. The only time when OpenDoors
- * is shutdown without going through this function should be as a result of
- * an explicit call to od_exit() by the client application.
- *
- * Parameters: btReasonForShutdown - An OpenDoors exit reason code.
- *
- * Return: Never returns.
- */
- void ODKrnlForceOpenDoorsShutdown(BYTE btReasonForShutdown)
- {
- BOOL bHangup;
- #ifdef OD_MULTITHREADED
- /* First, wait until an OpenDoors API is active. This way, we won't */
- /* interrupt any client application operations that may leave the */
- /* system in an unstable state (for instance, interrupting some file */
- /* I/O operations). */
- ODKrnlWaitForExclusiveControl();
- #endif /* OD_MULTITHREADED */
- bKernelActive = TRUE;
- /* Determine whether we should hangup on the user before exiting. */
- if(btReasonForShutdown == ERRORLEVEL_HANGUP
- || btReasonForShutdown == ERRORLEVEL_INACTIVITY)
- {
- bHangup = TRUE;
- }
- else
- {
- bHangup = FALSE;
- }
- /* Record exit reason in global variable. */
- btExitReason = btReasonForShutdown - 1;
- /* Use the client-defined errorlevel, if any. */
- if(od_control.od_errorlevel[0])
- {
- od_exit(od_control.od_errorlevel[btReasonForShutdown], bHangup);
- }
- /* Otherwise, use the default OpenDoors errorlevel. */
- else
- {
- od_exit(btReasonForShutdown - 1, bHangup);
- }
- }
- /* ========================================================================= */
- /* OpenDoors Kernel multithreaded implementation. */
- /* ========================================================================= */
- #ifdef OD_MULTITHREADED
- /* ----------------------------------------------------------------------------
- * ODKrnlRemoteInputThread() *** PRIVATE FUNCTION ***
- *
- * Code for the remote input thread. This thread executes an infinite loop,
- * blocking until a character is received from the remote system, and then
- * adding this character to the common local/remote input queue. This thread
- * should be given higher than normal priority.
- *
- * In non-multithreaded versions of OpenDoors, the task of checking for new
- * characters from the remote system and adding them to the common input
- * queue is performed on each call to od_kernel().
- *
- * Parameters: As dictated for any thread function.
- *
- * Return: As dictated for any thread function.
- */
- DWORD OD_THREAD_FUNC ODKrnlRemoteInputThread(void *pParam)
- {
- char chReceived;
- /* We keep looping until someone else terminates this thread. */
- for(;;)
- {
- /* Get next character from the modem, blocking if no character */
- /* is waiting. */
- ODComGetByte(hSerialPort, &chReceived, TRUE);
- /* Handle this received character, adding it to the local/remote */
- /* common input queue, if appropriate. */
- ODKrnlHandleReceivedChar(chReceived, TRUE);
- }
- return(0);
- }
- /* ----------------------------------------------------------------------------
- * ODKrnlNoCarrierThread() *** PRIVATE FUNCTION ***
- *
- * Thread which performs carrier detection. Normally, this thread doesn't
- * execute at all, but instead blocks waiting for a no carrier serial port
- * event. Only when the carrier detect signal goes low does this thread
- * execute, performing its one purpose in live - to trigger an OpenDoors
- * shutdown. This thread should be given higher than normal priority.
- *
- * This thread should only be created when OpenDoors is operating in remote
- * mode.
- *
- * In non-multithreaded versions of OpenDoors, this task is performed by
- * od_kernel().
- *
- * Parameters: As dictated for any thread function.
- *
- * Return: As dictated for any thread function.
- */
- DWORD OD_THREAD_FUNC ODKrnlNoCarrierThread(void *pParam)
- {
- /* Block until the carrier detect signal goes low with carrier */
- /* detection enabled. */
- for(;;)
- {
- /* Wait for carrier detect signal to go low. */
- ODComWaitEvent(hSerialPort, kNoCarrier);
- /* If carrier detection has not been disabled, then we have found */
- /* a condition where OpenDoors should exit. */
- if(!(od_control.od_disable&DIS_CARRIERDETECT)) break;
- /* If we have no carrier but carrier detection is currently */
- /* disabled, then we sleep for a while before checking again. */
- /* This isn't a very elegant implementation, and perhaps a */
- /* better approach will be used for future versions. */
- od_sleep(NO_CARRIER_THREAD_SLEEP_TIME);
- }
- /* Force OpenDoors to exit. */
- ODKrnlForceOpenDoorsShutdown(ERRORLEVEL_NOCARRIER);
- return(0);
- }
- /* ----------------------------------------------------------------------------
- * ODKrnlTimeUpdateThread() *** PRIVATE FUNCTION ***
- *
- * Thread which performs time limit updating and checking. This thread executes
- * an infinite loop, sleeping for several seconds, waking up to perform time
- * limit updating, and then going back to sleep. This thead should typically
- * operate at normal priority.
- *
- * In non-multithreaded versions of OpenDoors, this task is performed by
- * od_kernel().
- *
- * Parameters: As dictated for any thread function.
- *
- * Return: As dictated for any thread function.
- */
- DWORD OD_THREAD_FUNC ODKrnlTimeUpdateThread(void *pParam)
- {
- /* We keep looping until someone else terminates this thread. */
- for(;;)
- {
- /* Sleep until it is time to do the next update. */
- od_sleep(TIME_UPDATE_THREAD_SLEEP_TIME);
- /* Now, perform time update. */
- ODKrnlTimeUpdate();
- }
- return(0);
- }
- /* ----------------------------------------------------------------------------
- * ODKrnlWaitForExclusiveControl() *** PRIVATE FUNCTION ***
- *
- * Claims exclusive control of the application by the OpenDoors kernel. This is
- * required to ensure that the client application is not busy when the
- * OpenDoors kernel interrupts other operations for one reason or another
- * (for example, to start chat mode or to force the program to exit).
- *
- * Parameters: None
- *
- * Return: void
- */
- static void ODKrnlWaitForExclusiveControl(void)
- {
- /* If we already have exclusive control, then don't do anything. */
- if(bHaveExclusiveControl) return;
- /* Wait until an OpenDoors API is active. */
- ODSemaphoreDown(hODActiveSemaphore, OD_NO_TIMEOUT);
- /* Now, suspend the client thread. */
- ASSERT(hClientThread != NULL);
- ODThreadSuspend(hClientThread);
- /* Record that we now have exclusive control. */
- bHaveExclusiveControl = TRUE;
- }
- /* ----------------------------------------------------------------------------
- * ODKrnlGiveUpExclusiveControl() *** PRIVATE FUNCTION ***
- *
- * Relinguishes exclusive control of the application by the OpenDoors kernel.
- * A call to this function should only take place after a previous call to
- * ODKrnlWaitForExclusiveControl().
- *
- * Parameters: None
- *
- * Return: void
- */
- static void ODKrnlGiveUpExclusiveControl(void)
- {
- /* If we don't have exclusive control, then this call doesn't do */
- /* anything. */
- if(!bHaveExclusiveControl) return;
- /* First, restart the client thread. */
- ASSERT(hClientThread != NULL);
- ODThreadResume(hClientThread);
- /* Now, allow currently active OpenDoors API to return control */
- /* to the client application. */
- ODSemaphoreUp(hODActiveSemaphore, 1);
- /* Note that we no longer have exclusive control. */
- bHaveExclusiveControl = FALSE;
- }
- #endif /* OD_MULTITHREADED */
- /* ========================================================================= */
- /* OpenDoors chat mode. */
- /* ========================================================================= */
- BOOL bChatted;
- BOOL bSysopColor;
- #ifdef OD_MULTITHREADED
- /* ----------------------------------------------------------------------------
- * ODKrnlChatThread() *** PRIVATE FUNCTION ***
- *
- * Thread which implements sysop <-> remote user chat mode.
- *
- * Parameters: As dictated for any thread function.
- *
- * Return: As dictated for any thread function.
- */
- DWORD OD_THREAD_FUNC ODKrnlChatThread(void *pParam)
- {
- BOOL bTriggeredInsideOpenDoors = bChatActivatedInternally;
- /* The chat thread doesn't start up chat mode until the kernel has */
- /* exclusive control of the client application. */
- if(bTriggeredInsideOpenDoors)
- {
- ODKrnlWaitForExclusiveControl();
- }
- /* Now, execute the chat mode loop. */
- ODKrnlChatMode();
- /* If we get here, then we are responsible for relinguishing exclusive */
- /* control of the application. */
- if(bTriggeredInsideOpenDoors)
- {
- ODKrnlGiveUpExclusiveControl();
- }
- /* Exit the chat thread. */
- return(0);
- }
- /* ----------------------------------------------------------------------------
- * ODKrnlStartChatThread()
- *
- * Starts the chat mode thread.
- *
- * Parameters: bTriggeredInternally - TRUE if chat mode has been triggered
- * inside OpenDoors, or FALSE if it has
- * been triggered by a call to od_chat().
- *
- * Return: kODRCSuccess on success, or an error code on failure.
- */
- tODResult ODKrnlStartChatThread(BOOL bTriggeredInternally)
- {
- tODResult Result;
- bChatActivatedInternally = bTriggeredInternally;
- Result = ODThreadCreate(&hChatThread, ODKrnlChatThread, NULL);
- if(Result != kODRCSuccess)
- {
- return(Result);
- }
- /* If chat mode command has been chosen, then toggle chat */
- /* mode on or off. */
- od_control.od_chat_active = TRUE;
- #ifdef ODPLAT_WIN32
- /* Update the enabled and checked state of commands. */
- ODFrameUpdateCmdUI();
- #endif /* ODPLAT_WIN32 */
- return(kODRCSuccess);
- }
- #endif /* OD_MULTITHREADED */
- /* ----------------------------------------------------------------------------
- * ODKrnlEndChatMode()
- *
- * Forces chat mode to exit.
- *
- * Parameters: None
- *
- * Return: void
- */
- void ODKrnlEndChatMode(void)
- {
- #ifdef OD_MULTITHREADED
- /* Shutdown the chat thread. */
- ODThreadTerminate(hChatThread);
- /* Perform post-chat cleanup operations. */
- ODKrnlChatCleanup();
- #else /* !OD_MULTITHREADED */
- /* Turn off chat mode. */
- od_control.od_chat_active = FALSE;
- #endif /* !OD_MULTITHREADED */
- }
- /* ----------------------------------------------------------------------------
- * od_chat()
- *
- * Allows the client application to activate the line-by-line default chat
- * mode provided by OpenDoors, allowing the local sysop and remote user to
- * communicate with one another in real time.
- *
- * Parameters: none
- *
- * Return: void
- */
- ODAPIDEF void ODCALL od_chat(void)
- {
- /* Log function entry if running in trace mode. */
- TRACE(TRACE_API, "od_chat()");
- /* Set the main chat active flag in od_control. */
- od_control.od_chat_active = TRUE;
- /* Initialize OpenDoors if it hasn't already been done. */
- if(!bODInitialized) od_init();
- OD_API_ENTRY();
- #ifdef OD_MULTITHREADED
- /* In multithreaded versions of OpenDoors, od_chat() causes the chat */
- /* mode thread to be started, which in turn implements chat mode. */
- /* od_chat() only returns when this thread exits. */
- if(ODKrnlStartChatThread(FALSE) != kODRCSuccess)
- {
- od_control.od_error = ERR_GENERALFAILURE;
- OD_API_EXIT();
- }
- /* Now, wait for the chat thread to exit. */
- ODThreadWaitForExit(hChatThread);
- /* Now, note that the chat thread no longer exists. */
- hChatThread = NULL;
- #else /* !OD_MULTITHREADED */
- /* In non-multithreaded versions, a call to od_chat() maps directly to a */
- /* call to ODKrnlChatMode(), which implements chat mode. */
- ODKrnlChatMode();
- #endif /* !OD_MULTITHREADED */
- OD_API_EXIT();
- }
- /* ----------------------------------------------------------------------------
- * ODKrnlChatMode() *** PRIVATE FUNCTION ***
- *
- * Implements the OpenDoors chat mode.
- *
- * Parameters: None
- *
- * Return: void
- */
- static void ODKrnlChatMode(void)
- {
- BYTE chKeyPressed;
- char szCurrentWord[79];
- BYTE btWordLength = 0;
- BYTE btCurrentColumn = 0;
- char *pchCurrent;
- BYTE btCount;
- #ifndef OD_MULTITHREADED
- tODTimer Timer;
- #endif /* !OD_MULTITHREADED */
- /* Empty current word string. */
- szCurrentWord[0] = '\0';
- /* Save current display color attribute. */
- nChatOriginalAttrib = od_control.od_cur_attrib;
- /* Record that sysop has entered chat mode. */
- bChatted = TRUE;
- /* Turn off "user wants to chat" indicator, and force the status line. */
- /* to be updated. */
- od_control.user_wantchat = FALSE;
- #ifdef ODPLAT_WIN32
- ODFrameUpdateWantChat();
- #endif /* ODPLAT_WIN32 */
- bForceStatusUpdate = TRUE;
- CALL_KERNEL_IF_NEEDED();
- /* Note that chat mode is now active. */
- od_control.od_chat_active = TRUE;
- /* If a pre-chat function hook has been defined, then call it. */
- if(od_control.od_cbefore_chat!=NULL)
- {
- bShellChatActive = TRUE;
- (*od_control.od_cbefore_chat)();
- bShellChatActive = FALSE;
- /* If chat has been deactivated, then return right away */
- if(!od_control.od_chat_active) goto cleanup;
- }
- /* Display a message indicating that the sysop has entered chat mode. */
- od_set_attrib(od_control.od_chat_color1);
- if(od_control.od_before_chat != NULL)
- od_disp_str(od_control.od_before_chat);
- /* Currently set to sysop color. */
- bSysopColor = TRUE;
- /* If the logfile system is hooked up, then write a log entry */
- /* indicating that the sysop has entered chat mode. */
- if(pfLogWrite != NULL)
- {
- (*pfLogWrite)(9);
- }
- #ifndef OD_MULTITHREADED
- /* Start a timer that will elapse after 25 milliseconds. */
- ODTimerStart(&Timer, CHAT_YIELD_PERIOD);
- #endif /* !OD_MULTITHREADED */
- /* Loop while sysop chat mode is stilil on. */
- while(od_control.od_chat_active)
- {
- /* Obtain the next key from the user. */
- #ifdef OD_MULTITHREADED
- chKeyPressed = od_get_key(TRUE);
- #else /* !OD_MULTITHREADED */
- chKeyPressed = od_get_key(FALSE);
- #endif /* !OD_MULTITHREADED */
- /* If color not set correctly. */
- if((od_control.od_last_input && !bSysopColor)
- || (!od_control.od_last_input && bSysopColor))
- {
- /* If sysop was last person to type. */
- if(od_control.od_last_input)
- {
- /* Switch to sysop text color. */
- od_set_attrib(od_control.od_chat_color1);
- }
- else
- {
- /* Otherwise, switch to the user text color. */
- od_set_attrib(od_control.od_chat_color2);
- }
- /* Record current color setting. */
- bSysopColor = od_control.od_last_input;
- }
- /* If this is a displayable character. */
- if(chKeyPressed >= 32)
- {
- /* Display the character that was typed. */
- od_putch(chKeyPressed);
- /* If the user pressed spacebar, then this is the end of the */
- /* previous word. */
- if(chKeyPressed == 32)
- {
- btWordLength = 0;
- szCurrentWord[0] = 0;
- }
- /* Add this character to the current word, if we haven't exceeded */
- /* the maximum word length. */
- else if(btWordLength < 70)
- {
- szCurrentWord[btWordLength++] = chKeyPressed;
- szCurrentWord[btWordLength] = '\0';
- }
- /* If we are not yet at the end of the line, then increment the */
- /* current column number. */
- if(btCurrentColumn < 75)
- {
- ++btCurrentColumn;
- }
- /* If we are at the end of the line. */
- else
- {
- /* If the current word should be wrapped to the next line. */
- if(btWordLength < 70 && btWordLength > 0)
- {
- /* Generate a string to erase the word from the current line. */
- pchCurrent = (char *)szODWorkString;
- for(btCount = 0; btCount < btWordLength; ++btCount)
- {
- *(pchCurrent++) = 8;
- }
- for(btCount = 0; btCount < btWordLength; ++btCount)
- {
- *(pchCurrent++) = ' ';
- }
- *pchCurrent = '\0';
- /* Display the string to erase the old word. */
- od_disp_str(szODWorkString);
- /* Move to the next line. */
- od_disp_str("\n\r");
- /* Redisplay the word on the next line. */
- od_disp_str(szCurrentWord);
- /* Update current column number. */
- btCurrentColumn = btWordLength;
- }
- /* If we have reached the end of the line, but word wrap should */
- /* not be performed. */
- else
- {
- /* Move to the next line. */
- od_disp_str("\n\r");
- /* Update the current column number. */
- btCurrentColumn = 0;
- }
- /* Reset the current word information. */
- btWordLength = 0;
- szCurrentWord[0] = 0;
- }
- }
- /* If the backspace key was pressed. */
- else if(chKeyPressed == 8)
- {
- /* Send backspace sequence. */
- od_disp_str(szBackspaceWithDelete);
- /* If we are in the middle of a word, then we must remove the */
- /* last character of the word. */
- if(btWordLength > 0)
- {
- szCurrentWord[--btWordLength] = '\0';
- }
- /* Update the current column number. */
- if(btCurrentColumn > 0) --btCurrentColumn;
- }
- /* If the enter key was pressed. */
- else if(chKeyPressed == 13)
- {
- /* Send carriage return / line feed sequence. */
- od_disp_str("\n\r");
- /* Reset the current word contents. */
- btWordLength = 0;
- szCurrentWord[0] = 0;
- /* Update the current column number. */
- btCurrentColumn = 0;
- }
- /* If the sysop pressed the escape key. */
- else if(chKeyPressed == 27 && od_control.od_last_input)
- {
- /* Exit chat mode. */
- goto cleanup;
- }
- #ifndef OD_MULTITHREADED
- /* Give up processor after 25 milliseconds elapsed. */
- else if(ODTimerElapsed(&Timer))
- {
- od_sleep(0);
- /* Restart the timer, so that it will elapse after another */
- /* 25 milliseconds. */
- ODTimerStart(&Timer, CHAT_YIELD_PERIOD);
- }
- #endif /* !OD_MULTITHREADED */
- }
- cleanup:
- ODKrnlChatCleanup();
- }
- /* ----------------------------------------------------------------------------
- * ODKrnlChatCleanup() *** PRIVATE FUNCTION ***
- *
- * Performs post-chat operations, such as resetting the original display
- * color, etc.
- *
- * Parameters: None
- *
- * Return: void
- */
- static void ODKrnlChatCleanup(void)
- {
- od_set_attrib(od_control.od_chat_color1);
- /* Indicate that chat mode is exiting. */
- if(od_control.od_after_chat != NULL)
- {
- od_disp_str(od_control.od_after_chat);
- }
- /* If an after chat function has been provided, then call it. */
- if(od_control.od_cafter_chat != NULL)
- {
- bShellChatActive = TRUE;
- (*od_control.od_cafter_chat)();
- bShellChatActive = FALSE;
- }
- /* If the logfile system is hooked up, then write a line to the log */
- /* indicating that chat mode has been exited. */
- if(pfLogWrite != NULL)
- {
- (*pfLogWrite)(10);
- }
- /* Restore original display color attribute. */
- od_set_attrib(nChatOriginalAttrib);
- /* Record that chat mode is no longer active. */
- od_control.od_chat_active = FALSE;
- #ifdef ODPLAT_WIN32
- /* Update the enabled and checked state of commands. */
- ODFrameUpdateCmdUI();
- #endif /* ODPLAT_WIN32 */
- #ifdef OD_MULTITHREADED
- if(bChatActivatedInternally)
- {
- ODKrnlGiveUpExclusiveControl();
- }
- #endif
- }
- #ifdef ODPLAT_NIX
- #ifdef USE_KERNEL_SIGNAL
- /* ----------------------------------------------------------------------------
- * sig_run_kernel(sig) *** PRIVATE FUNCTION ***
- *
- * Runs od_kernel() on a SIGALRM
- *
- */
- static void sig_run_kernel(int sig)
- {
- od_kernel();
- }
- /* ----------------------------------------------------------------------------
- * sig_run_kernel(sig) *** PRIVATE FUNCTION ***
- *
- * Runs od_kernel() on a SIGALRM
- *
- */
- static void sig_get_char(int sig)
- {
- static char ch;
- /* Loop, obtaining any new characters from the serial port and */
- /* adding them to the common local/remote input queue. */
- while(ODComGetByte(hSerialPort, &ch, FALSE) == kODRCSuccess)
- {
- ODKrnlHandleReceivedChar(ch, TRUE);
- }
- }
- static void sig_no_carrier(int sig)
- {
- if(od_control.baud != 0 && )
- {
- if(!(od_control.od_disable&DIS_CARRIERDETECT))
- ODKrnlForceOpenDoorsShutdown(ERRORLEVEL_NOCARRIER);
- }
- }
- #endif
- #endif
|