ds2.9a12/bin/
ds2.9a12/extra/
ds2.9a12/extra/crat/
ds2.9a12/extra/creremote/
ds2.9a12/extra/mingw/
ds2.9a12/extra/wolfpaw/
ds2.9a12/fluffos-2.14-ds13/
ds2.9a12/fluffos-2.14-ds13/Win32/
ds2.9a12/fluffos-2.14-ds13/compat/
ds2.9a12/fluffos-2.14-ds13/compat/simuls/
ds2.9a12/fluffos-2.14-ds13/include/
ds2.9a12/fluffos-2.14-ds13/testsuite/
ds2.9a12/fluffos-2.14-ds13/testsuite/clone/
ds2.9a12/fluffos-2.14-ds13/testsuite/command/
ds2.9a12/fluffos-2.14-ds13/testsuite/data/
ds2.9a12/fluffos-2.14-ds13/testsuite/etc/
ds2.9a12/fluffos-2.14-ds13/testsuite/include/
ds2.9a12/fluffos-2.14-ds13/testsuite/inherit/
ds2.9a12/fluffos-2.14-ds13/testsuite/inherit/master/
ds2.9a12/fluffos-2.14-ds13/testsuite/log/
ds2.9a12/fluffos-2.14-ds13/testsuite/single/
ds2.9a12/fluffos-2.14-ds13/testsuite/single/tests/compiler/
ds2.9a12/fluffos-2.14-ds13/testsuite/single/tests/efuns/
ds2.9a12/fluffos-2.14-ds13/testsuite/single/tests/operators/
ds2.9a12/fluffos-2.14-ds13/testsuite/u/
ds2.9a12/lib/cmds/admins/
ds2.9a12/lib/cmds/common/
ds2.9a12/lib/cmds/creators/include/
ds2.9a12/lib/daemon/services/
ds2.9a12/lib/daemon/tmp/
ds2.9a12/lib/doc/
ds2.9a12/lib/doc/bguide/
ds2.9a12/lib/doc/efun/all/
ds2.9a12/lib/doc/efun/arrays/
ds2.9a12/lib/doc/efun/buffers/
ds2.9a12/lib/doc/efun/compile/
ds2.9a12/lib/doc/efun/floats/
ds2.9a12/lib/doc/efun/functions/
ds2.9a12/lib/doc/efun/general/
ds2.9a12/lib/doc/efun/mixed/
ds2.9a12/lib/doc/efun/numbers/
ds2.9a12/lib/doc/efun/parsing/
ds2.9a12/lib/doc/hbook/
ds2.9a12/lib/doc/help/classes/
ds2.9a12/lib/doc/help/races/
ds2.9a12/lib/doc/lfun/
ds2.9a12/lib/doc/lfun/all/
ds2.9a12/lib/doc/lfun/lib/abilities/
ds2.9a12/lib/doc/lfun/lib/armor/
ds2.9a12/lib/doc/lfun/lib/bank/
ds2.9a12/lib/doc/lfun/lib/bot/
ds2.9a12/lib/doc/lfun/lib/clay/
ds2.9a12/lib/doc/lfun/lib/clean/
ds2.9a12/lib/doc/lfun/lib/clerk/
ds2.9a12/lib/doc/lfun/lib/client/
ds2.9a12/lib/doc/lfun/lib/combat/
ds2.9a12/lib/doc/lfun/lib/connect/
ds2.9a12/lib/doc/lfun/lib/container/
ds2.9a12/lib/doc/lfun/lib/corpse/
ds2.9a12/lib/doc/lfun/lib/creator/
ds2.9a12/lib/doc/lfun/lib/daemon/
ds2.9a12/lib/doc/lfun/lib/damage/
ds2.9a12/lib/doc/lfun/lib/deterioration/
ds2.9a12/lib/doc/lfun/lib/donate/
ds2.9a12/lib/doc/lfun/lib/door/
ds2.9a12/lib/doc/lfun/lib/equip/
ds2.9a12/lib/doc/lfun/lib/file/
ds2.9a12/lib/doc/lfun/lib/fish/
ds2.9a12/lib/doc/lfun/lib/fishing/
ds2.9a12/lib/doc/lfun/lib/flashlight/
ds2.9a12/lib/doc/lfun/lib/follow/
ds2.9a12/lib/doc/lfun/lib/ftp_client/
ds2.9a12/lib/doc/lfun/lib/ftp_data_connection/
ds2.9a12/lib/doc/lfun/lib/fuel/
ds2.9a12/lib/doc/lfun/lib/furnace/
ds2.9a12/lib/doc/lfun/lib/genetics/
ds2.9a12/lib/doc/lfun/lib/holder/
ds2.9a12/lib/doc/lfun/lib/id/
ds2.9a12/lib/doc/lfun/lib/interactive/
ds2.9a12/lib/doc/lfun/lib/lamp/
ds2.9a12/lib/doc/lfun/lib/leader/
ds2.9a12/lib/doc/lfun/lib/light/
ds2.9a12/lib/doc/lfun/lib/limb/
ds2.9a12/lib/doc/lfun/lib/living/
ds2.9a12/lib/doc/lfun/lib/load/
ds2.9a12/lib/doc/lfun/lib/look/
ds2.9a12/lib/doc/lfun/lib/manipulate/
ds2.9a12/lib/doc/lfun/lib/meal/
ds2.9a12/lib/doc/lfun/lib/messages/
ds2.9a12/lib/doc/lfun/lib/player/
ds2.9a12/lib/doc/lfun/lib/poison/
ds2.9a12/lib/doc/lfun/lib/position/
ds2.9a12/lib/doc/lfun/lib/post_office/
ds2.9a12/lib/doc/lfun/lib/potion/
ds2.9a12/lib/doc/lfun/lib/room/
ds2.9a12/lib/doc/lfun/lib/server/
ds2.9a12/lib/doc/lfun/lib/spell/
ds2.9a12/lib/doc/lfun/lib/torch/
ds2.9a12/lib/doc/lfun/lib/vendor/
ds2.9a12/lib/doc/lfun/lib/virt_sky/
ds2.9a12/lib/doc/lfun/lib/weapon/
ds2.9a12/lib/doc/lfun/lib/worn_storage/
ds2.9a12/lib/doc/lpc/basic/
ds2.9a12/lib/doc/lpc/concepts/
ds2.9a12/lib/doc/lpc/constructs/
ds2.9a12/lib/doc/lpc/etc/
ds2.9a12/lib/doc/lpc/intermediate/
ds2.9a12/lib/doc/lpc/types/
ds2.9a12/lib/doc/misc/
ds2.9a12/lib/doc/old/
ds2.9a12/lib/domains/
ds2.9a12/lib/domains/Praxis/adm/
ds2.9a12/lib/domains/Praxis/attic/
ds2.9a12/lib/domains/Praxis/cemetery/mon/
ds2.9a12/lib/domains/Praxis/data/
ds2.9a12/lib/domains/Praxis/death/
ds2.9a12/lib/domains/Praxis/mountains/
ds2.9a12/lib/domains/Praxis/obj/armour/
ds2.9a12/lib/domains/Praxis/obj/magic/
ds2.9a12/lib/domains/Praxis/obj/weapon/
ds2.9a12/lib/domains/Praxis/orc_valley/
ds2.9a12/lib/domains/Ylsrim/
ds2.9a12/lib/domains/Ylsrim/adm/
ds2.9a12/lib/domains/Ylsrim/armor/
ds2.9a12/lib/domains/Ylsrim/broken/
ds2.9a12/lib/domains/Ylsrim/fish/
ds2.9a12/lib/domains/Ylsrim/meal/
ds2.9a12/lib/domains/Ylsrim/npc/
ds2.9a12/lib/domains/Ylsrim/obj/
ds2.9a12/lib/domains/Ylsrim/virtual/
ds2.9a12/lib/domains/Ylsrim/weapon/
ds2.9a12/lib/domains/campus/adm/
ds2.9a12/lib/domains/campus/etc/
ds2.9a12/lib/domains/campus/meals/
ds2.9a12/lib/domains/campus/save/
ds2.9a12/lib/domains/campus/txt/ai/charles/
ds2.9a12/lib/domains/campus/txt/ai/charles/bak2/
ds2.9a12/lib/domains/campus/txt/ai/charles/bak2/bak1/
ds2.9a12/lib/domains/campus/txt/ai/charly/
ds2.9a12/lib/domains/campus/txt/ai/charly/bak/
ds2.9a12/lib/domains/campus/txt/jenny/
ds2.9a12/lib/domains/cave/doors/
ds2.9a12/lib/domains/cave/etc/
ds2.9a12/lib/domains/cave/meals/
ds2.9a12/lib/domains/cave/weap/
ds2.9a12/lib/domains/default/creator/
ds2.9a12/lib/domains/default/doors/
ds2.9a12/lib/domains/default/etc/
ds2.9a12/lib/domains/default/vehicles/
ds2.9a12/lib/domains/default/virtual/
ds2.9a12/lib/domains/default/weap/
ds2.9a12/lib/domains/town/txt/shame/
ds2.9a12/lib/domains/town/virtual/
ds2.9a12/lib/domains/town/virtual/bottom/
ds2.9a12/lib/domains/town/virtual/space/
ds2.9a12/lib/estates/
ds2.9a12/lib/ftp/
ds2.9a12/lib/lib/comp/
ds2.9a12/lib/lib/daemons/
ds2.9a12/lib/lib/daemons/include/
ds2.9a12/lib/lib/lvs/
ds2.9a12/lib/lib/user/
ds2.9a12/lib/lib/virtual/
ds2.9a12/lib/log/
ds2.9a12/lib/log/adm/
ds2.9a12/lib/log/archive/
ds2.9a12/lib/log/chan/
ds2.9a12/lib/log/errors/
ds2.9a12/lib/log/law/adm/
ds2.9a12/lib/log/law/email/
ds2.9a12/lib/log/law/names/
ds2.9a12/lib/log/law/sites-misc/
ds2.9a12/lib/log/law/sites-register/
ds2.9a12/lib/log/law/sites-tempban/
ds2.9a12/lib/log/law/sites-watch/
ds2.9a12/lib/log/open/
ds2.9a12/lib/log/reports/
ds2.9a12/lib/log/router/
ds2.9a12/lib/log/secure/
ds2.9a12/lib/log/watch/
ds2.9a12/lib/obj/book_source/
ds2.9a12/lib/obj/include/
ds2.9a12/lib/powers/prayers/
ds2.9a12/lib/powers/spells/
ds2.9a12/lib/realms/template/adm/
ds2.9a12/lib/realms/template/area/armor/
ds2.9a12/lib/realms/template/area/npc/
ds2.9a12/lib/realms/template/area/obj/
ds2.9a12/lib/realms/template/area/room/
ds2.9a12/lib/realms/template/area/weap/
ds2.9a12/lib/realms/template/bak/
ds2.9a12/lib/realms/template/cmds/
ds2.9a12/lib/save/kills/o/
ds2.9a12/lib/secure/cfg/classes/
ds2.9a12/lib/secure/cmds/builders/
ds2.9a12/lib/secure/cmds/creators/include/
ds2.9a12/lib/secure/cmds/players/
ds2.9a12/lib/secure/cmds/players/include/
ds2.9a12/lib/secure/daemon/imc2server/
ds2.9a12/lib/secure/daemon/include/
ds2.9a12/lib/secure/lib/
ds2.9a12/lib/secure/lib/include/
ds2.9a12/lib/secure/lib/net/include/
ds2.9a12/lib/secure/lib/std/
ds2.9a12/lib/secure/log/adm/
ds2.9a12/lib/secure/log/bak/
ds2.9a12/lib/secure/log/intermud/
ds2.9a12/lib/secure/log/network/
ds2.9a12/lib/secure/modules/
ds2.9a12/lib/secure/npc/
ds2.9a12/lib/secure/obj/include/
ds2.9a12/lib/secure/room/
ds2.9a12/lib/secure/save/
ds2.9a12/lib/secure/save/backup/
ds2.9a12/lib/secure/save/boards/
ds2.9a12/lib/secure/tmp/
ds2.9a12/lib/secure/upgrades/files/
ds2.9a12/lib/secure/verbs/creators/
ds2.9a12/lib/std/board/
ds2.9a12/lib/std/lib/
ds2.9a12/lib/tmp/
ds2.9a12/lib/verbs/admins/include/
ds2.9a12/lib/verbs/builders/
ds2.9a12/lib/verbs/common/
ds2.9a12/lib/verbs/common/include/
ds2.9a12/lib/verbs/creators/
ds2.9a12/lib/verbs/creators/include/
ds2.9a12/lib/verbs/rooms/
ds2.9a12/lib/verbs/rooms/include/
ds2.9a12/lib/www/client/
ds2.9a12/lib/www/errors/
ds2.9a12/lib/www/images/
ds2.9a12/lib/www/lpmuds/downloads_files/
ds2.9a12/lib/www/lpmuds/intermud_files/
ds2.9a12/lib/www/lpmuds/links_files/
ds2.9a12/win32/
/*    /lib/net/server.c
 *    From the Dead Souls LPC Library
 *    a TCP server object
 *    created by Descartes of Borg 950429
 *    Version: @(#) server.c 1.3@(#)
 *    Last modified: 96/12/19
 */

#include <lib.h>
#include <daemons.h>
#include <network.h>
#include <runtime_config.h>
#include <message_class.h>

inherit LIB_DAEMON;
string mcolor;
int mclass;
int ftp_port = PORT_FTP;

mapping ServerMap = ([]);
mapping Listen = ([]);

private static int          DestructOnClose= 0;
private static int          MaxBytes       = get_config(__MAX_BYTE_TRANSFER__);
private static int          Port           = 0;
private static string       SocketObject   = 0;
private static int          SocketType     = STREAM;
private static mapping      Sockets        = ([]);

static void eventSocketError(string msg, int code);

/* ******************* server.c attributes ************************ */
int GetDestructOnClose() {
    return DestructOnClose;
}

static int SetDestructOnClose(int x) {
    return (DestructOnClose = x);
}

int GetSocketType() {
    return SocketType;
}

static int SetSocketType(int x ) {
    return (SocketType = x);
}

/* ******************** server.c events *************************** */
static int eventClose(mixed sock) {
    mapping s; 

    trr("LIB_SERVER: eventClose trying to close: "+identify(sock),mcolor,mclass);
    if(mapp(sock)) {
        s = copy(sock);
    }
    if( intp(sock) ) {
        s = copy(Sockets[sock]);
    }
    else if( objectp(sock) ) {
        s = copy(Sockets[sock->GetDescriptor()]);
    }
    if( !s ) {
        return 0;
    }
    if( s["Blocking"] ) {
        s["Closing"] = 1;
        return 1;
    }
    if( Sockets[s["Descriptor"]] ) {
        map_delete(Sockets, s["Descriptor"]);
    }
    socket_close(s["Descriptor"]);
    if( s["Owner"] ) {
        s["Owner"]->eventSocketClosed();
    }
    if( DestructOnClose && sock == Listen ) {
        Destruct();
    }
    sock = 0;
}

int eventCreateSocket(int port) {
    int x;
    Listen["Blocking"] = 0; /* servers are not blocking to start */
    x = socket_create(SocketType,
      "eventServerReadCallback", 
      "eventServerAbortCallback");
    if( x < 0 ) {
        eventSocketError("Error in socket_create().", x);
        return x;
    }
    Listen["Descriptor"] = x;
    x = socket_bind(Listen["Descriptor"], port);
    if( x != EESUCCESS ) {
        eventClose(Listen);
        eventSocketError("Error in socket_bind().", x);
        return x;
    }
    x = socket_listen(Listen["Descriptor"], "eventServerListenCallback");
    if( x != EESUCCESS ) {
        eventClose(Listen);
        eventSocketError("Error in socket_listen().", x);
        return x;
    }
    trr("LIB_SERVER: eventCreateSocket, port: "+port+", x: "+x,mcolor,mclass);
}

static int Destruct() {
    if( daemon::Destruct() ) {
        foreach(int fd, mapping socket in Sockets) {
            trr("server:Destruct: fd: "+fd+", "+socket_address(fd),mcolor,mclass);
            if(socket && socket["Owner"])
                socket["Owner"]->eventShutdown();
        }
        eventClose(Listen);
        return 1;
    }
    else {
        return 0;
    }
}

int eventDestruct() {
    if( !master()->valid_apply(({})) ) {
        return 0;
    }
    return daemon::eventDestruct();
}

static void eventNewConnection(object socket) {
    int fd = socket->GetDescriptor();
    if(!Sockets[fd]) Sockets[fd] = ([]);
    trr("LIB_SERVER: eventNewConnection, socket: "+identify(socket),mcolor,mclass);
    trr("LIB_SERVER: eventNewConnection, socket->GetDescriptor(): "+identify(socket->GetDescriptor()),mcolor,mclass);
    Sockets[fd]["Descriptor"] = socket->GetDescriptor();
    Sockets[fd]["Blocking"] = 0;
    Sockets[fd]["Owner"] = socket;
    socket->StartService(); // added for welcome
}

static void eventServerAbortCallback(int fd) {
    trr("server:eventServerAbortCallback: fd: "+fd+", "+socket_address(fd),mcolor,mclass);
    eventClose(fd);
}

int eventShutdown() {
    if( previous_object() != find_object(INET_D) ) {
        return 0;
    }
    Destruct();
    return 1;
}

static void eventServerListenCallback(int fd) {
    int x;

    trr("server:eventServerListenCallback: fd: "+fd+", "+socket_address(fd),mcolor,mclass);
    x = socket_accept(fd,
      "eventServerReadCallback", 
      "eventServerWriteCallback");
    if( x < 0 ) {
        trr("Error in socket_accept().",mcolor,mclass);
        eventSocketError("Error in socket_accept().", x);
        return;
    }
    if( Sockets[x] ) {
        eventClose(Sockets[x]);
    }
    trr("LIB_SERVER SocketObject: "+SocketObject,mcolor,mclass);
    eventNewConnection(new(SocketObject, x, this_object()));
}

static void eventServerReadCallback(int fd, mixed val) {
    trr("server:eventServerReadCallback: fd: "+fd+", "+socket_address(fd),mcolor,mclass);
    trr("server: I think that Sockets["+fd+"] is: "+identify(Sockets[fd]),mcolor,mclass);
    if( !Sockets[fd] || !Sockets[fd]["Owner"] ) {
        trr("server: No owner found for this data.",mcolor,mclass);
        eventClose(fd);
        return;
    }
    else {
        trr("Sockets["+fd+"]: "+identify(Sockets[fd]),mcolor,mclass);
        trr("sizeof(val): "+sizeof(val),mcolor,mclass);
        trr("typeof(val): "+typeof(val),mcolor,mclass);
        if(bufferp(val)) trr("  val: "+identify(read_buffer(val)),mcolor,mclass);
        else trr("  val: "+identify(val),mcolor,mclass);
        Sockets[fd]["Owner"]->eventRead(val);
    }
}

static void eventServerWriteCallback(int fd) {
    int x;
    mapping sock;

    trr("server:eventServerWriteCallback: fd: "+fd+", "+socket_address(fd),mcolor,mclass);
    if( Listen && Listen["Descriptor"] == fd ) {
        sock = Listen;
    }
    else if( Sockets[fd] ) {
        sock = Sockets[fd];
    }
    else {
        return;
    }
    sock["Blocking"] = 0;
    if( !sock["Buffer"] && sock["Closing"] ) {
        eventClose(sock);
        return;
    }
    x = EESUCCESS;
    while( sock["Buffer"] && x == EESUCCESS ) {
        switch( x = socket_write(sock["Descriptor"], sock["Buffer"][0]) ) {
        case EESUCCESS:
            break;
        case EECALLBACK:
            sock["Blocking"] = 1;
            break;
        case EEWOULDBLOCK: 
            call_out( (: eventServerWriteCallback :), 0, fd);
            return;
        case EEALREADY:
            sock["Blocking"] = 1;
            return;
        default:
            eventClose(sock);
            eventSocketError("Error in socket_write().", x);
            return;
        }
        if( sizeof(sock["Buffer"]) == 1 ) {
            sock["Buffer"] = 0;
            if( sock["Closing"] && !sock["Blocking"] ) {
                eventClose(sock);
            }
        }
        else {
            sock["Buffer"] = sock["Buffer"][1..];
        }
    }
}

static void eventSocketError(string msg, int code) {
    log_file("servers", "Error code: " + code + "\n" + msg + "\n");
    trr("LIB_SERVER Error code: " + code + "\n" + msg + "\n","red",mclass);
}

varargs int eventWrite(object owner, mixed val, int close) {
    int fd = owner->GetDescriptor();
    mapping sock;

    trr("server:eventWrite: fd: "+fd+", "+socket_address(fd),mcolor,mclass);
    if(bufferp(val)) trr("       eventWrite: owner: "+identify(owner)+", val: "+identify(read_buffer(val)),mcolor,mclass);
    else trr("       eventWrite: owner: "+identify(owner)+", val: "+identify(val),mcolor,mclass);
    trr("       eventWrite: close: "+close,mcolor,mclass);

    if( Listen && Listen["Descriptor"] == fd ) {
        sock = Listen;
    }
    else if( Sockets[fd] ) {
        sock = Sockets[fd];
    }
    else {
        return 0;
    }
    if( owner != sock["Owner"] ) {
        return 0;
    }
    if( SocketType != STREAM || stringp(val)) {
        if( sock["Buffer"] ) {
            sock["Buffer"] += ({ val });
        }
        else {
            sock["Buffer"] = ({ val });
        }
    }
    else {
        buffer data = val;
        int size = sizeof(data);
        int count = (size/MaxBytes) + 1;

        if( !sock["Buffer"] ) {
            sock["Buffer"] = ({});
        }
        for(int i=0; i<count; i++) {
            int length, ptr;
            buffer b;

            ptr = count * MaxBytes;
            if( size - ptr > MaxBytes ) {
                length = MaxBytes;
            }
            else {
                length = size - ptr;
            }
            b = read_buffer(data, ptr, length);
            sock["Buffer"] = ({ sock["Buffer"]..., b });
        }
    }

    sock["Closing"] = close;
    if( !sock["Blocking"] ) {
        eventServerWriteCallback(sock["Descriptor"]);
    }
    return 1;
}

/* ******************** server.c driver applies ******************* */
varargs static void create(int port, int type, string socket_obj) {
    daemon::create();
    SetNoClean(1);

    if(port == PORT_FTP){mcolor="green";mclass=MSG_FTP;}
    if(port == PORT_HFTP){mcolor="white";mclass=MSG_HFTP;}
    else if(port == PORT_HTTP){mcolor="cyan";mclass=MSG_HTTP;}
    else if(port == PORT_RCP){mcolor="yellow";mclass=MSG_RCP;}
    else if(port == PORT_OOB){mcolor="red";mclass=MSG_OOB;}
    else { mcolor="blue",mclass=MSG_CONV;}

    if( socket_obj ) {
        SocketObject = socket_obj;
    }
    if( !undefinedp(type) ) {
        SocketType = type;
    }
    if( !undefinedp(port) ) {
        Port = port;
        call_out((: eventCreateSocket :), 1, Port);
    }
}