/
MudOS_0.9.19/bin/
MudOS_0.9.19/doc/concepts/
MudOS_0.9.19/doc/driver/
MudOS_0.9.19/doc/efuns/bitstrings/
MudOS_0.9.19/doc/efuns/buffers/
MudOS_0.9.19/doc/efuns/communication/
MudOS_0.9.19/doc/efuns/core/
MudOS_0.9.19/doc/efuns/mappings/
MudOS_0.9.19/doc/efuns/math/
MudOS_0.9.19/doc/efuns/security/
MudOS_0.9.19/doc/lpc/constructs/
MudOS_0.9.19/doc/lpc/types/
MudOS_0.9.19/doc/platforms/
MudOS_0.9.19/etc/
MudOS_0.9.19/mudlib/
MudOS_0.9.19/mudlib/lil/
MudOS_0.9.19/mudlib/lil/clone/
MudOS_0.9.19/mudlib/lil/command/
MudOS_0.9.19/mudlib/lil/data/
MudOS_0.9.19/mudlib/lil/etc/
MudOS_0.9.19/mudlib/lil/include/
MudOS_0.9.19/mudlib/lil/inherit/
MudOS_0.9.19/mudlib/lil/inherit/master/
MudOS_0.9.19/mudlib/lil/log/
MudOS_0.9.19/mudlib/lil/single/
MudOS_0.9.19/mudlib/lil/u/
MudOS_0.9.19/src/testsuite/
MudOS_0.9.19/src/testsuite/clone/
MudOS_0.9.19/src/testsuite/command/
MudOS_0.9.19/src/testsuite/data/
MudOS_0.9.19/src/testsuite/etc/
MudOS_0.9.19/src/testsuite/include/
MudOS_0.9.19/src/testsuite/inherit/
MudOS_0.9.19/src/testsuite/inherit/master/
MudOS_0.9.19/src/testsuite/log/
MudOS_0.9.19/src/testsuite/single/
MudOS_0.9.19/src/testsuite/single/efuns/
MudOS_0.9.19/src/testsuite/u/
/*
	efuns_sock.c: this file contains the socket efunctions called from
	inside eval_instruction() in interpret.c.
*/

#include "efuns.h"
#include "stralloc.h"
#if defined(__386BSD__) || defined(SunOS_5)
#include <unistd.h>
#endif

#ifdef F_SOCKET_CREATE
void
f_socket_create(num_arg, instruction)
    int num_arg, instruction;
{
    int fd;
    struct svalue *arg, *ret;
    struct vector *info;

    arg = sp - num_arg + 1;
	if ((num_arg == 3) && (arg[2].type != T_STRING)) {
		bad_arg(3, instruction);
	}

    info = allocate_array(4);
    info->ref--;
    info->item[0].type = T_NUMBER;
    info->item[0].u.number = -1;
    assign_socket_owner(info->item + 1, current_object);
    info->item[2].type = T_STRING;
    info->item[2].subtype = STRING_CONSTANT;
    info->item[2].u.string = "N/A";
    info->item[3].type = T_NUMBER;
    info->item[3].u.number = -1;

    push_object(current_object);
    push_string("create", STRING_CONSTANT);
    push_vector(info);

    ret = apply_master_ob("valid_socket", 3);
    if (!IS_ZERO(ret)) {
        if (num_arg == 2)
            fd = socket_create(arg[0].u.number, arg[1].u.string, NULL);
        else {
            fd = socket_create(arg[0].u.number, arg[1].u.string, arg[2].u.string);
        }
        pop_n_elems(num_arg); /* pop both args off stack    */
        push_number(fd); /* push return int onto stack */
    } else {
        pop_n_elems(num_arg); /* pop both args off stack    */
        push_number(EESECURITY); /* Security violation attempted */
    }
}
#endif

#ifdef F_SOCKET_BIND
void
f_socket_bind(num_arg, instruction)
    int num_arg, instruction;
{
    int i, fd, port;
    struct svalue *ret;
    struct vector *info;
    char addr[ADDR_BUF_SIZE];

    fd = (sp-1)->u.number;
    get_socket_address(fd, addr, &port);

    info = allocate_array(4);
    info->ref--;
    info->item[0].type = T_NUMBER;
    info->item[0].u.number = fd;
    assign_socket_owner(&info->item[1], get_socket_owner(fd));
    info->item[2].type = T_STRING;
    info->item[2].subtype = STRING_CONSTANT;
    info->item[2].u.string = addr;
    info->item[3].type = T_NUMBER;
    info->item[3].u.number = port;

    push_object(current_object);
    push_string("bind", STRING_CONSTANT);
    push_vector(info);

    ret = apply_master_ob("valid_socket", 3);
    if (!IS_ZERO(ret)) {
	i = socket_bind(fd, sp->u.number);
	pop_n_elems(2); /* pop both args off stack    */
	push_number(i); /* push return int onto stack */
    } else {
	pop_n_elems(2); /* pop both args off stack    */
	push_number(EESECURITY); /* Security violation attempted */
    }
}
#endif
  
#ifdef F_SOCKET_LISTEN
void
f_socket_listen(num_arg, instruction)
     int num_arg, instruction;
{
    int i, fd, port;
    struct svalue *ret;
    struct vector *info;
    char addr[ADDR_BUF_SIZE];

    fd = (sp-1)->u.number;
    get_socket_address(fd, addr, &port);

    info = allocate_array(4);
    info->ref--;
    info->item[0].type = T_NUMBER;
    info->item[0].u.number = fd;
    assign_socket_owner(&info->item[1], get_socket_owner(fd));
    info->item[2].type = T_STRING;
    info->item[2].subtype = STRING_CONSTANT;
    info->item[2].u.string = addr;
    info->item[3].type = T_NUMBER;
    info->item[3].u.number = port;

    push_object(current_object);
    push_string("listen",STRING_CONSTANT);
    push_vector(info);

    ret = apply_master_ob("valid_socket",3);
    if (!IS_ZERO(ret)) {
	i = socket_listen(fd, sp->u.string);
	pop_n_elems(2); /* pop both args off stack    */
	push_number(i); /* push return int onto stack */
    } else {
	pop_n_elems(2); /* pop both args off stack    */
	push_number(EESECURITY); /* Security violation attempted */
    }
}
#endif

#ifdef F_SOCKET_ACCEPT
void
f_socket_accept(num_arg, instruction)
    int num_arg, instruction;
{
    int i, fd, port;
    struct svalue *ret;
    struct vector *info;
    char addr[ADDR_BUF_SIZE];
  
	if (sp->type != T_STRING) {
		bad_arg(3, instruction);
	}
    fd = (sp-2)->u.number;
    get_socket_address(fd, addr, &port);

    info = allocate_array(4);
    info->ref--;
    info->item[0].type = T_NUMBER;
    info->item[0].u.number = fd;
    assign_socket_owner(&info->item[1], get_socket_owner(fd));
    info->item[2].type = T_STRING;
    info->item[2].subtype = STRING_CONSTANT;
    info->item[2].u.string = addr;
    info->item[3].type = T_NUMBER;
    info->item[3].u.number = port;

    push_object(current_object);
    push_string("accept", STRING_CONSTANT);
    push_vector(info);

    ret = apply_master_ob("valid_socket", 3);
    if (!IS_ZERO(ret)) {
	i = socket_accept(fd, (sp-1)->u.string, sp->u.string);
	pop_n_elems(3); /* pop both args off stack    */
	push_number(i); /* push return int onto stack */
    } else {
	pop_n_elems(3); /* pop both args off stack    */
	push_number(EESECURITY); /* Security violation attempted */
    }
}
#endif

#ifdef F_SOCKET_CONNECT
void
f_socket_connect(num_arg, instruction)
     int num_arg, instruction;
{
    int i, fd, port;
    struct svalue *ret;
    struct vector *info;
    char addr[ADDR_BUF_SIZE];

	if ((sp-1)->type != T_STRING) {
		bad_arg(3, instruction);
	}
	if (sp->type != T_STRING) {
		bad_arg(4, instruction);
	}
    fd = (sp-3)->u.number;
    get_socket_address(fd, addr, &port);

    info = allocate_array(4);
    info->ref--;
    info->item[0].type = T_NUMBER;
    info->item[0].u.number = fd;
    assign_socket_owner(&info->item[1], get_socket_owner(fd));
    info->item[2].type = T_STRING;
    info->item[2].subtype = STRING_CONSTANT;
    info->item[2].u.string = addr;
    info->item[3].type = T_NUMBER;
    info->item[3].u.number = port;

    push_object(current_object);
    push_string("connect", STRING_CONSTANT);
    push_vector(info);

    ret = apply_master_ob("valid_socket", 3);
    if (!IS_ZERO(ret)) {
	i = socket_connect(fd, (sp-2)->u.string, (sp-1)->u.string,
		sp->u.string);
	pop_n_elems(4); /* pop all args off stack     */
	push_number(i); /* push return int onto stack */
    } else {
	pop_n_elems(4); /* pop all args off stack     */
	push_number(EESECURITY); /* Security violation attempted */
    }
}
#endif

#ifdef F_SOCKET_WRITE
void
f_socket_write(num_arg, instruction)
    int num_arg, instruction;
{
    int i, fd, port;
    struct svalue *arg, *ret;
    struct vector *info;
    char addr[ADDR_BUF_SIZE];

    arg = sp - num_arg + 1;
    if ((num_arg == 3) && (arg[2].type != T_STRING))
    {
            bad_arg(3, instruction);
    }
    fd = arg[0].u.number;
    get_socket_address(fd, addr, &port);

    info = allocate_array(4);
    info->ref--;
    info->item[0].type = T_NUMBER;
    info->item[0].u.number = fd;
    assign_socket_owner(&info->item[1], get_socket_owner(fd));
    info->item[2].type = T_STRING;
    info->item[2].subtype = STRING_CONSTANT;
    info->item[2].u.string = addr;
    info->item[3].type = T_NUMBER;
    info->item[3].u.number = port;

    push_object(current_object);
    push_string("write", STRING_CONSTANT);
    push_vector(info);

    ret = apply_master_ob("valid_socket", 3);
    if (!IS_ZERO(ret)) {
	i = socket_write(fd, &arg[1],
		(num_arg == 3) ? arg[2].u.string : (char *)NULL);
	pop_n_elems(num_arg); /* pop both args off stack    */
	push_number(i); /* push return int onto stack */
    } else {
	pop_n_elems(num_arg); /* pop both args off stack    */
	push_number(EESECURITY); /* Security violation attempted */
    }
}
#endif

#ifdef F_SOCKET_CLOSE
void
f_socket_close(num_arg, instruction)
    int num_arg, instruction;
{
    int i, fd, port;
    struct svalue *ret;
    struct vector *info;
    char addr[ADDR_BUF_SIZE];

    fd = sp->u.number;
    get_socket_address(fd, addr, &port);

    info = allocate_array(4);
    info->ref--;
    info->item[0].type = T_NUMBER;
    info->item[0].u.number = fd;
    assign_socket_owner(&info->item[1], get_socket_owner(fd));
    info->item[2].type = T_STRING;
    info->item[2].subtype = STRING_CONSTANT;
    info->item[2].u.string = addr;
    info->item[3].type = T_NUMBER;
    info->item[3].u.number = port;

    push_object(current_object);
    push_string("close", STRING_CONSTANT);
    push_vector(info);

    ret = apply_master_ob("valid_socket", 3);
    if (!IS_ZERO(ret)) {
	i = socket_close(fd);
	pop_stack();    /* pop int arg off stack      */
	push_number(i); /* push return int onto stack */
    } else { 
	pop_stack();    /* pop int arg off stack      */
	push_number(EESECURITY); /* Security violation attempted */
    }
}
#endif

#ifdef F_SOCKET_RELEASE
void
f_socket_release(num_arg, instruction)
    int num_arg, instruction;
{
    int i, fd, port;
    struct svalue *ret;
    struct vector *info;
    char addr[ADDR_BUF_SIZE];

	if (sp->type != T_STRING) {
		bad_arg(3, instruction);
	}
    fd = (sp-2)->u.number;
    get_socket_address(fd, addr, &port);

    info = allocate_array(4);
    info->ref--;
    info->item[0].type = T_NUMBER;
    info->item[0].u.number = fd;
    assign_socket_owner(&info->item[1], get_socket_owner(fd));
    info->item[2].type = T_STRING;
    info->item[2].subtype = STRING_CONSTANT;
    info->item[2].u.string = addr;
    info->item[3].type = T_NUMBER;
    info->item[3].u.number = port;

    push_object(current_object);
    push_string("release", STRING_CONSTANT);
    push_vector(info);

    ret = apply_master_ob("valid_socket", 3);
    if (!IS_ZERO(ret)) {
	i = socket_release((sp-2)->u.number, (sp-1)->u.ob, sp->u.string);
	pop_n_elems(3); /* pop all args off stack     */
	push_number(i); /* push return int onto stack */
    } else {
	pop_n_elems(3); /* pop all args off stack     */
	push_number(EESECURITY); /* Security violation attempted */
    }
}
#endif

#ifdef F_SOCKET_ACQUIRE
void
f_socket_acquire(num_arg, instruction)
    int num_arg, instruction;
{
    int i, fd, port;
    struct svalue *ret;
    struct vector *info;
    char addr[ADDR_BUF_SIZE];

	if ((sp-1)->type != T_STRING) {
		bad_arg(3, instruction);
	}
	if (sp->type != T_STRING) {
		bad_arg(4, instruction);
	}
    fd = (sp-1)->u.number;
    get_socket_address(fd, addr, &port);

    info = allocate_array(4);
    info->ref--;
    info->item[0].type = T_NUMBER;
    info->item[0].u.number = fd;
    assign_socket_owner(&info->item[1], get_socket_owner(fd));
    info->item[2].type = T_STRING;
    info->item[2].subtype = STRING_CONSTANT;
    info->item[2].u.string = addr;
    info->item[3].type = T_NUMBER;
    info->item[3].u.number = port;

    push_object(current_object);
    push_string("acquire", STRING_CONSTANT);
    push_vector(info);

    ret = apply_master_ob("valid_socket", 3);
    if (!IS_ZERO(ret)) {
	i = socket_acquire((sp-3)->u.number, (sp-2)->u.string,
		(sp-1)->u.string, sp->u.string);
	pop_n_elems(4); /* pop both args off stack    */
	push_number(i); /* push return int onto stack */
    } else {
	pop_n_elems(4); /* pop both args off stack    */
	push_number(EESECURITY); /* Security violation attempted */
    }
}
#endif

#ifdef F_SOCKET_ERROR
void
f_socket_error(num_arg, instruction)
    int num_arg, instruction;
{
    char *error;

    error = socket_error(sp->u.number);
    pop_stack();    /* pop int arg off stack      */
    push_string(error, STRING_CONSTANT); /* push return string onto stack */
}
#endif

#ifdef F_SOCKET_ADDRESS
void
f_socket_address(num_arg, instruction)
    int num_arg, instruction;
{
    char *str;
    int port;
    char addr[ADDR_BUF_SIZE];
 
/*
 * Ok, we will add in a cute little check thing here to see if it is
 * an object or not...
 */ 
    if (sp->type == T_OBJECT) {
      char *tmp;
/* This is so we can get the address of interactives as well. */

      if (!sp->u.ob->interactive) {
        pop_stack();
        push_null();
        return ;
      }
      tmp = inet_ntoa(sp->u.ob->interactive->addr.sin_addr);
      str = (char *)DMALLOC(strlen(tmp) + 5 + 3, 33, "f_socket_address");
      sprintf(str, "%s %d", tmp, sp->u.ob->interactive->addr.sin_port);
      pop_stack();
      push_string(str, STRING_MALLOC);
      return ;
    }
    get_socket_address(sp->u.number, addr, &port);
    str = (char *)DMALLOC(strlen(addr) + 5 + 3, 33, "f_socket_address");
    sprintf(str, "%s %d", addr, port);
    pop_stack();
    push_string(str, STRING_MALLOC);
} /* f_socket_address() */
#endif

#ifdef F_DUMP_SOCKET_STATUS
void
f_dump_socket_status(num_arg, instruction)
     int num_arg, instruction;
{
  dump_socket_status();
  push_number(0);
}
#endif