Skip to content
Snippets Groups Projects
  • Henrik (Grubba) Grubbström's avatar
    b788cdfc
    Moved include of global.h first. · b788cdfc
    Henrik (Grubba) Grubbström authored
    Rev: src/modules/MIME/mime.c:1.12
    Rev: src/modules/Msql/msqlmod.c:1.8
    Rev: src/modules/Mysql/mysql.c:1.23
    Rev: src/modules/Mysql/result.c:1.12
    Rev: src/modules/Odbc/odbc.c:1.10
    Rev: src/modules/Odbc/odbc_result.c:1.14
    Rev: src/modules/Oracle/oracle.c:1.14
    Rev: src/modules/Postgres/pgresult.c:1.7
    Rev: src/modules/Postgres/postgres.c:1.7
    Rev: src/modules/Regexp/glue.c:1.12
    Rev: src/modules/Ssleay/ssleay.c:1.9
    Rev: src/modules/_Image_JPEG/image_jpeg.c:1.19
    Rev: src/modules/math/math.c:1.12
    Rev: src/modules/system/passwords.c:1.15
    Rev: src/modules/system/syslog.c:1.6
    Rev: src/modules/system/system.c:1.56
    Rev: src/opcodes.c:1.29
    Rev: src/operators.c:1.37
    b788cdfc
    History
    Moved include of global.h first.
    Henrik (Grubba) Grubbström authored
    Rev: src/modules/MIME/mime.c:1.12
    Rev: src/modules/Msql/msqlmod.c:1.8
    Rev: src/modules/Mysql/mysql.c:1.23
    Rev: src/modules/Mysql/result.c:1.12
    Rev: src/modules/Odbc/odbc.c:1.10
    Rev: src/modules/Odbc/odbc_result.c:1.14
    Rev: src/modules/Oracle/oracle.c:1.14
    Rev: src/modules/Postgres/pgresult.c:1.7
    Rev: src/modules/Postgres/postgres.c:1.7
    Rev: src/modules/Regexp/glue.c:1.12
    Rev: src/modules/Ssleay/ssleay.c:1.9
    Rev: src/modules/_Image_JPEG/image_jpeg.c:1.19
    Rev: src/modules/math/math.c:1.12
    Rev: src/modules/system/passwords.c:1.15
    Rev: src/modules/system/syslog.c:1.6
    Rev: src/modules/system/system.c:1.56
    Rev: src/opcodes.c:1.29
    Rev: src/operators.c:1.37
opcodes.c 21.85 KiB
/*\
||| This file a part of Pike, and is copyright by Fredrik Hubinette
||| Pike is distributed as GPL (General Public License)
||| See the files COPYING and DISCLAIMER for more information.
\*/
#include "global.h"
#include <math.h>
#include <ctype.h>
#include "interpret.h"
#include "svalue.h"
#include "array.h"
#include "stralloc.h"
#include "mapping.h"
#include "multiset.h"
#include "opcodes.h"
#include "object.h"
#include "error.h"
#include "pike_types.h"
#include "pike_memory.h"
#include "fd_control.h"
#include "cyclic.h"
#include "builtin_functions.h"
#include "module_support.h"

RCSID("$Id: opcodes.c,v 1.29 1998/07/04 16:56:14 grubba Exp $");

void index_no_free(struct svalue *to,struct svalue *what,struct svalue *ind)
{
  INT32 i;
  switch(what->type)
  {
  case T_ARRAY:
    simple_array_index_no_free(to,what->u.array,ind);
    break;

  case T_MAPPING:
    mapping_index_no_free(to,what->u.mapping,ind);
    break;

  case T_OBJECT:
    object_index_no_free(to, what->u.object, ind);
    break;

  case T_MULTISET:
    i=multiset_member(what->u.multiset, ind);
    to->type=T_INT;
    to->subtype=i ? NUMBER_UNDEFINED : 0;
    to->u.integer=i;
    break;

  case T_STRING:
    if(ind->type==T_INT)
    {
      i=ind->u.integer;
      if(i<0)
	i+=what->u.string->len;
      if(i<0 || i>=what->u.string->len)
	error("Index %d is out of range 0 - %d.\n", i, what->u.string->len-1);
      else
	i=EXTRACT_UCHAR(what->u.string->str + i);
      to->type=T_INT;
      to->subtype=NUMBER_NUMBER;
      to->u.integer=i;
      break;
    }else{
      error("Index is not an integer.\n");
    }

  case T_PROGRAM:
    program_index_no_free(to, what->u.program, ind);
    break;

  case T_FUNCTION:
    {
      struct program *p = program_from_svalue(what);
      if (p) {
	program_index_no_free(to, p, ind);
	break;
      }
    }
    /* FALL THROUGH */

  default:
    error("Indexing a basic type.\n");
  }
}

void o_index(void)
{
  struct svalue s;
  index_no_free(&s,sp-2,sp-1);
  pop_n_elems(2);
  *sp=s;
  sp++;
}

void o_cast(struct pike_string *type, INT32 run_time_type)
{
  INT32 i;

  if(run_time_type != sp[-1].type)
  {
    if(run_time_type == T_MIXED)
      return;

    if(sp[-1].type == T_OBJECT)
    {
      struct pike_string *s;
      s=describe_type(type);
      push_string(s);
      if(!sp[-2].u.object->prog)
	error("Cast called on destructed object.\n");
      if(FIND_LFUN(sp[-2].u.object->prog,LFUN_CAST) == -1)
	error("No cast method in object.\n");
      apply_lfun(sp[-2].u.object, LFUN_CAST, 1);
      free_svalue(sp-2);
      sp[-2]=sp[-1];
      sp--;
      return;
    }

    switch(run_time_type)
    {
      case T_MIXED:
	return;
	
      case T_ARRAY:
	switch(sp[-1].type)
	{
	  case T_MAPPING:
	  {
	    struct array *a=mapping_to_array(sp[-1].u.mapping);
	    pop_stack();
	    push_array(a);
	    break;
	  }

	  case T_STRING:
	    f_values(1);
	    break;

	  case T_MULTISET:
	    f_indices(1);
	    break;

	  default:
	    error("Cannot cast %s to array.\n",get_name_of_type(sp[-1].type));
	      
	}
	break;
	
      case T_INT:
	switch(sp[-1].type)
	{
	  case T_FLOAT:
	    i=(int)(sp[-1].u.float_number);
	    break;
	    
	  case T_STRING:
	    i=STRTOL(sp[-1].u.string->str,0,0);
	    free_string(sp[-1].u.string);
	    break;
	    
	  default:
	    error("Cannot cast %s to int.\n",get_name_of_type(sp[-1].type));
	}
	
	sp[-1].type=T_INT;
	sp[-1].u.integer=i;
	break;
	
      case T_FLOAT:
      {
	FLOAT_TYPE f;
	switch(sp[-1].type)
	{
	  case T_INT:
	    f=(FLOAT_TYPE)(sp[-1].u.integer);
	    break;
	    
	  case T_STRING:
	    f=STRTOD(sp[-1].u.string->str,0);
	    free_string(sp[-1].u.string);
	    break;
	    
	  default:
	    error("Cannot cast %s to float.\n",get_name_of_type(sp[-1].type));
	    f=0.0;
	}
	
	sp[-1].type=T_FLOAT;
	sp[-1].u.float_number=f;
	break;
      }
      
      case T_STRING:
      {
	char buf[200];
	switch(sp[-1].type)
	{
	  case T_INT:
	    sprintf(buf,"%ld",(long)sp[-1].u.integer);
	    break;
	    
	  case T_FLOAT:
	    sprintf(buf,"%f",(double)sp[-1].u.float_number);
	    break;

	  case T_ARRAY:
	    {
	      int i;
	      struct array *a = sp[-1].u.array;
	      struct pike_string *s;

	      for(i = a->size; i--; ) {
		if (a->item[i].type != T_INT) {
		  error("cast: Item %d is not an integer.\n", i);
		}
		if ((a->item[i].u.integer < 0) ||
		    (a->item[i].u.integer > 255)) {
		  error("cast: Wide strings are not supported yet.\n"
			"Index %d is %d, and is out of range 0 .. 255.\n",
			i, a->item[i].u.integer);
		}
	      }
	      s = begin_shared_string(a->size);
	      for(i = a->size; i--; ) {
		s->str[i] = a->item[i].u.integer;
	      }
	      s = end_shared_string(s);
	      pop_stack();
	      push_string(s);
	      return;
	    }
	    break;
	    
	  default:
	    error("Cannot cast %s to string.\n",get_name_of_type(sp[-1].type));
	}
	
	sp[-1].type=T_STRING;
	sp[-1].u.string=make_shared_string(buf);
	break;
      }
      
      case T_OBJECT:
	switch(sp[-1].type)
	{
	  case T_STRING:
	    if(fp->pc)
	    {
	      INT32 lineno;
	      push_text(get_line(fp->pc, fp->context.prog, &lineno));
	    }else{
	      push_int(0);
	    }
	    APPLY_MASTER("cast_to_object",2);
	    return;
	    
	  case T_FUNCTION:
	    sp[-1].type = T_OBJECT;
	    break;

	  default:
	    error("Cannot cast %s to object.\n",get_name_of_type(sp[-1].type));
	}
	break;
	
      case T_PROGRAM:
      switch(sp[-1].type)
      {
	case T_STRING:
	  if(fp->pc)
	  {
	    INT32 lineno;
	    push_text(get_line(fp->pc, fp->context.prog, &lineno));
	  }else{
	    push_int(0);
	  }
	  APPLY_MASTER("cast_to_program",2);
	  return;
	  
	case T_FUNCTION:
	{
	  struct program *p=program_from_function(sp-1);
	  if(p)
	  {
	    add_ref(p);
	    pop_stack();
	    push_program(p);
	  }else{
	    pop_stack();
	    push_int(0);
	  }
	}
	return;

	default:
	  error("Cannot cast %s to a program.\n",get_name_of_type(sp[-1].type));
      }
    }
  }

#ifdef DEBUG
  if(run_time_type != sp[-1].type)
    fatal("Internal error: Cast failed (run_time_type = %d, sp[-1].type = %d.)\n",run_time_type,sp[-1].type);
#endif

  switch(run_time_type)
  {
    case T_ARRAY:
    {
      struct pike_string *itype;
      INT32 run_time_itype;

      push_string(itype=index_type(type,0));
      run_time_itype=compile_type_to_runtime_type(itype);

      if(run_time_itype != T_MIXED)
      {
	struct array *a;
	struct array *tmp=sp[-2].u.array;
	DECLARE_CYCLIC();
	
	if((a=(struct array *)BEGIN_CYCLIC(tmp,0)))
	{
	  ref_push_array(a);
	}else{
	  INT32 e,i;
	  struct pike_string *s;
#ifdef DEBUG
	  struct svalue *save_sp=sp+1;
#endif
	  push_array(a=allocate_array(tmp->size));
	  SET_CYCLIC_RET(a);
	  
	  for(e=0;e<a->size;e++)
	  {
	    push_svalue(tmp->item+e);
	    o_cast(itype, run_time_itype);
	    array_set_index(a,e,sp-1);
	    pop_stack();
	  }
#ifdef DEBUG
	  if(save_sp!=sp)
	    fatal("o_cast left stack droppings.\n");
#endif
	  END_CYCLIC();
	}
	assign_svalue(sp-3,sp-1);
	pop_stack();
      }
      pop_stack();
    }
  }
}


void f_cast(void)
{
#ifdef DEBUG
  struct svalue *save_sp=sp;
#endif
  o_cast(sp[-2].u.string,
	 compile_type_to_runtime_type(sp[-2].u.string));
#ifdef DEBUG
  if(save_sp != sp)
    fatal("Internal error: o_cast() left droppings on stack.\n");
#endif
  free_svalue(sp-2);
  sp[-2]=sp[-1];
  sp--;
}


/*
  flags:
   *
  operators:
  %d
  %s
  %f
  %c
  %n
  %[
  %%
*/

static int read_set(unsigned char *match,int cnt,char *set,int match_len)
{
  int init;
  int last=0;
  int e;

  if(cnt>=match_len)
    error("Error in sscanf format string.\n");

  if(match[cnt]=='^')
  {
    for(e=0;e<256;e++) set[e]=1;
    init=0;
    cnt++;
    if(cnt>=match_len)
      error("Error in sscanf format string.\n");
  }else{
    for(e=0;e<256;e++) set[e]=0;
    init=1;
  }
  if(match[cnt]==']' || match[cnt]=='-')
  {
    set[last=match[cnt]]=init;
    cnt++;
    if(cnt>=match_len)
      error("Error in sscanf format string.\n");
  }

  for(;match[cnt]!=']';cnt++)
  {
    if(match[cnt]=='-')
    {
      cnt++;
      if(cnt>=match_len)
	error("Error in sscanf format string.\n");
      if(match[cnt]==']')
      {
	set['-']=init;
	break;
      }
      for(e=last;e<(int) EXTRACT_UCHAR(match+cnt);e++) set[e]=init;
    }
    set[last=EXTRACT_UCHAR(match+cnt)]=init;
  }
  return cnt;
}



/* Parse binary IEEE strings on a machine which uses a different kind
   of floating point internally */

#ifndef FLOAT_IS_IEEE_BIG
#ifndef FLOAT_IS_IEEE_LITTLE
#define NEED_CUSTOM_IEEE
#endif
#endif
#ifndef NEED_CUSTOM_IEEE
#ifndef DOUBLE_IS_IEEE_BIG
#ifndef DOUBLE_IS_IEEE_LITTLE
#define NEED_CUSTOM_IEEE
#endif
#endif
#endif

#ifdef NEED_CUSTOM_IEEE

#if HAVE_LDEXP
#define LDEXP ldexp
#else
extern double LDEXP(double x, int exp); /* defined in encode.c */
#endif

INLINE static float low_parse_IEEE_float(char *b, int sz)
{
  unsigned INT32 f, extra_f;
  int s, e;
  unsigned char x[4];
  double r;

  x[0] = EXTRACT_UCHAR(b);
  x[1] = EXTRACT_UCHAR(b+1);
  x[2] = EXTRACT_UCHAR(b+2);
  x[3] = EXTRACT_UCHAR(b+3);
  s = ((x[0]&0x80)? 1 : 0);

  if(sz==4) {
    e = (((int)(x[0]&0x7f))<<1)|((x[1]&0x80)>>7);
    f = (((unsigned INT32)(x[1]&0x7f))<<16)|(((unsigned INT32)x[2])<<8)|x[3];
    extra_f = 0;
    if(e==255)
      e = 9999;
    else if(e>0) {
      f |= 0x00800000;
      e -= 127+23;
    } else
      e -= 126+23;
  } else {
    e = (((int)(x[0]&0x7f))<<4)|((x[1]&0xf0)>>4);
    f = (((unsigned INT32)(x[1]&0x0f))<<16)|(((unsigned INT32)x[2])<<8)|x[3];
    extra_f = (((unsigned INT32)EXTRACT_UCHAR(b+4))<<24)|
      (((unsigned INT32)EXTRACT_UCHAR(b+5))<<16)|
      (((unsigned INT32)EXTRACT_UCHAR(b+6))<<8)|
      ((unsigned INT32)EXTRACT_UCHAR(b+7));
    if(e==2047)
      e = 9999;
    else if(e>0) {
      f |= 0x00100000;
      e -= 1023+20;
    } else
      e -= 1022+20;
  }
  if(e>=9999)
    if(f||extra_f) {
      /* NAN */
      
      /* Hmm...  No idea how to generate NaN in a portable way. */
      /* Let's turn it into a 0 for now... */
      return (float)0.0;
    } else {
      /* +/- Infinity */
#ifdef HUGE_VAL
      return (float)(s? -HUGE_VAL:HUGE_VAL);
#else
      /* This number is infinite enough...  :) */
      e = 1024;
      f = 1;
      extra_f = 0;
#endif
    }

  r = (double)f;
  if(extra_f)
    r += ((double)extra_f)/4294967296.0;
  return (float)(s? -LDEXP(r, e):LDEXP(r, e));
}

#endif

static INT32 really_low_sscanf(char *input,
			       long input_len,
			       char *match,
			       long match_len,
			       long *chars_matched,
			       int *success)
{
  struct svalue sval;
  int e,cnt,matches,eye,arg;
  int no_assign,field_length;
  char set[256];
  struct svalue *argp;
  
  success[0]=0;

  arg=eye=matches=0;

  for(cnt=0;cnt<match_len;cnt++)
  {
    for(;cnt<match_len;cnt++)
    {
      if(match[cnt]=='%')
      {
        if(match[cnt+1]=='%')
        {
          cnt++;
        }else{
          break;
        }
      }
      if(eye>=input_len || input[eye]!=match[cnt])
      {
	chars_matched[0]=eye;
	return matches;
      }
      eye++;
    }
    if(cnt>=match_len)
    {
      chars_matched[0]=eye;
      return matches;
    }

#ifdef DEBUG
    if(match[cnt]!='%' || match[cnt+1]=='%')
    {
      fatal("Error in sscanf.\n");
    }
#endif

    no_assign=0;
    field_length=-1;

    cnt++;
    if(cnt>=match_len)
      error("Error in sscanf format string.\n");

    while(1)
    {
      switch(match[cnt])
      {
      case '*':
	no_assign=1;
	cnt++;
	if(cnt>=match_len)
	  error("Error in sscanf format string.\n");
	continue;

      case '0': case '1': case '2': case '3': case '4':
      case '5': case '6': case '7': case '8': case '9':
      {
	char *t;
	field_length=STRTOL(match+cnt,&t,10);
	cnt=t-match;
	continue;
      }

	case '{':
	{
	  ONERROR err;
	  long tmp;
	  for(e=cnt+1,tmp=1;tmp;e++)
	  {
	    if(!match[e])
	    {
	      error("Missing %%} in format string.\n");
	      break;		/* UNREACHED */
	    }
	    if(match[e]=='%')
	    {
	      switch(match[e+1])
	      {
		case '%': e++; break;
		case '}': tmp--; break;
		case '{': tmp++; break;
	      }
	    }
	  }
	  sval.type=T_ARRAY;
	  sval.u.array=allocate_array(0);
	  SET_ONERROR(err, do_free_array, sval.u.array);

	  while(1)
	  {
	    int yes;
	    struct svalue *save_sp=sp;
	    really_low_sscanf(input+eye,
			      input_len-eye,
			      match+cnt+1,
			      e-cnt-2,
			      &tmp,
			      &yes);
	    if(yes)
	    {
	      f_aggregate(sp-save_sp);
	      sval.u.array=append_array(sval.u.array,sp-1);
	      pop_stack();
	      eye+=tmp;
	    }else{
	      pop_n_elems(sp-save_sp);
	      break;
	    }
	  }
	  cnt=e;
	  UNSET_ONERROR(err);
	  break;
	}

      case 'c':
	if(field_length == -1) field_length = 1;
	if(eye+field_length > input_len)
	{
	  chars_matched[0]=eye;
	  return matches;
	}
	sval.type=T_INT;
	sval.subtype=NUMBER_NUMBER;
	sval.u.integer=0;
	while(--field_length >= 0)
	{
	  sval.u.integer<<=8;
	  sval.u.integer|=EXTRACT_UCHAR(input+eye);
	  eye++;
	}
	break;

      case 'd':
      {
	char * t;

	if(eye>=input_len)
	{
	  chars_matched[0]=eye;
	  return matches;
	}
	if(field_length != -1 && eye+field_length < input_len)
	{
	  char save=input[eye+field_length];
	  input[eye+field_length]=0; /* DANGEROUS */
	  sval.u.integer=STRTOL(input+eye,&t,10);
	  input[eye+field_length]=save;
	}else
	  sval.u.integer=STRTOL(input+eye,&t,10);

	if(input + eye == t)
	{
	  chars_matched[0]=eye;
	  return matches;
	}
	eye=t-input;
	sval.type=T_INT;
	sval.subtype=NUMBER_NUMBER;
	break;
      }

      case 'x':
      {
	char * t;

	if(eye>=input_len)
	{
	  chars_matched[0]=eye;
	  return matches;
	}
	if(field_length != -1 && eye+field_length < input_len)
	{
	  char save=input[eye+field_length];
	  input[eye+field_length]=0; /* DANGEROUS */
	  sval.u.integer=STRTOL(input+eye,&t,16);
	  input[eye+field_length]=save;
	}else
	  sval.u.integer=STRTOL(input+eye,&t,16);
	if(input + eye == t)
	{
	  chars_matched[0]=eye;
	  return matches;
	}
	eye=t-input;
	sval.type=T_INT;
	sval.subtype=NUMBER_NUMBER;
	break;
      }

      case 'o':
      {
	char * t;

	if(eye>=input_len)
	{
	  chars_matched[0]=eye;
	  return matches;
	}
	if(field_length != -1 && eye+field_length < input_len)
	{
	  char save=input[eye+field_length];
	  input[eye+field_length]=0; /* DANGEROUS */
	  sval.u.integer=STRTOL(input+eye,&t,8);
	  input[eye+field_length]=save;
	}else
	  sval.u.integer=STRTOL(input+eye,&t,8);
	if(input + eye == t)
	{
	  chars_matched[0]=eye;
	  return matches;
	}
	eye=t-input;
	sval.type=T_INT;
	sval.subtype=NUMBER_NUMBER;
	break;
      }

      case 'D':
      case 'i':
      {
	char * t;

	if(eye>=input_len)
	{
	  chars_matched[0]=eye;
	  return matches;
	}
	if(field_length != -1 && eye+field_length < input_len)
	{
	  char save=input[eye+field_length];
	  input[eye+field_length]=0; /* DANGEROUS */
	  sval.u.integer=STRTOL(input+eye,&t,0);
	  input[eye+field_length]=save;
	}else
	  sval.u.integer=STRTOL(input+eye,&t,0);
	if(input + eye == t)
	{
	  chars_matched[0]=eye;
	  return matches;
	}
	eye=t-input;
	sval.type=T_INT;
	sval.subtype=NUMBER_NUMBER;
	break;
      }

      case 'f':
      {
	char * t;

	if(eye>=input_len)
	{
	  chars_matched[0]=eye;
	  return matches;
	}
	sval.u.float_number=STRTOD(input+eye,&t);
	if(input + eye == t)
	{
	  chars_matched[0]=eye;
	  return matches;
	}
	eye=t-input;
	sval.type=T_FLOAT;
#ifdef __CHECKER__
	sval.subtype=0;
#endif
	break;
      }

      case 'F':
	if(field_length == -1) field_length = 4;
	if(field_length != 4 && field_length != 8)
	  error("Invalid IEEE width %d in sscanf format string.\n",
		field_length);
	if(eye+field_length > input_len)
	{
	  chars_matched[0]=eye;
	  return matches;
	}
	sval.type=T_FLOAT;
#ifdef __CHECKER__
	sval.subtype=0;
#endif
	switch(field_length) {
	case 4:
#ifdef FLOAT_IS_IEEE_BIG
	  {
	    float f;
	    ((char *)&f)[0] = *(input+eye);
	    ((char *)&f)[1] = *(input+eye+1);
	    ((char *)&f)[2] = *(input+eye+2);
	    ((char *)&f)[3] = *(input+eye+3);
	    sval.u.float_number = f;
	  }
#else
#ifdef FLOAT_IS_IEEE_LITTLE
	  {
	    float f;
	    ((char *)&f)[3] = *(input+eye);
	    ((char *)&f)[2] = *(input+eye+1);
	    ((char *)&f)[1] = *(input+eye+2);
	    ((char *)&f)[0] = *(input+eye+3);
	    sval.u.float_number = f;
	  }
#else
	  sval.u.float_number = low_parse_IEEE_float(input+eye, 4);
#endif
#endif
	  eye += 4;
	  break;
	case 8:
#ifdef DOUBLE_IS_IEEE_BIG
	  {
	    double d;
	    ((char *)&d)[0] = *(input+eye);
	    ((char *)&d)[1] = *(input+eye+1);
	    ((char *)&d)[2] = *(input+eye+2);
	    ((char *)&d)[3] = *(input+eye+3);
	    ((char *)&d)[4] = *(input+eye+4);
	    ((char *)&d)[5] = *(input+eye+5);
	    ((char *)&d)[6] = *(input+eye+6);
	    ((char *)&d)[7] = *(input+eye+7);
	    sval.u.float_number = (float)d;
	  }
#else
#ifdef DOUBLE_IS_IEEE_LITTLE
	  {
	    double d;
	    ((char *)&d)[7] = *(input+eye);
	    ((char *)&d)[6] = *(input+eye+1);
	    ((char *)&d)[5] = *(input+eye+2);
	    ((char *)&d)[4] = *(input+eye+3);
	    ((char *)&d)[3] = *(input+eye+4);
	    ((char *)&d)[2] = *(input+eye+5);
	    ((char *)&d)[1] = *(input+eye+6);
	    ((char *)&d)[0] = *(input+eye+7);
	    sval.u.float_number = (float)d;
	  }
#else
	  sval.u.float_number = low_parse_IEEE_float(input+eye, 8);
#endif
#endif
	  eye += 8;
	  break;
	}
	break;

      case 's':
	if(field_length != -1)
	{
	  if(input_len - eye < field_length)
	  {
	    chars_matched[0]=eye;
	    return matches;
	  }

	  sval.type=T_STRING;
#ifdef __CHECKER__
	  sval.subtype=0;
#endif
	  sval.u.string=make_shared_binary_string(input+eye,field_length);
	  eye+=field_length;
	  break;
	}

	if(cnt+1>=match_len)
	{
	  sval.type=T_STRING;
#ifdef __CHECKER__
	  sval.subtype=0;
#endif
	  sval.u.string=make_shared_binary_string(input+eye,input_len-eye);
	  eye=input_len;
	  break;
	}else{
	  char *end_str_start;
	  char *end_str_end;
	  char *s=0;		/* make gcc happy */
	  char *p=0;		/* make gcc happy */
	  int start,contains_percent_percent, new_eye;

	  start=eye;
	  end_str_start=match+cnt+1;
          
	  s=match+cnt+1;
	test_again:
	  if(*s=='%')
	  {
	    s++;
	    if(*s=='*') s++;
	    switch(*s)
	    {
	    case 'n':
	      s++;
	      goto test_again;
	      
	    case 's':
	      error("Illegal to have two adjecent %%s.\n");
	      return 0;		/* make gcc happy */
	      
	      /* sscanf("foo-bar","%s%d",a,b) might not work as expected */
	    case 'd':
	      for(e=0;e<256;e++) set[e]=1;
	      for(e='0';e<='9';e++) set[e]=0;
	      set['-']=0;
	      goto match_set;

	    case 'o':
	      for(e=0;e<256;e++) set[e]=1;
	      for(e='0';e<='7';e++) set[e]=0;
	      goto match_set;

	    case 'x':
	      for(e=0;e<256;e++) set[e]=1;
	      for(e='0';e<='9';e++) set[e]=0;
	      for(e='a';e<='f';e++) set[e]=0;
	      goto match_set;

	    case 'D':
	      for(e=0;e<256;e++) set[e]=1;
	      for(e='0';e<='9';e++) set[e]=0;
	      set['-']=0;
	      set['x']=0;
	      goto match_set;

	    case 'f':
	      for(e=0;e<256;e++) set[e]=1;
	      for(e='0';e<='9';e++) set[e]=0;
	      set['.']=set['-']=0;
	      goto match_set;

	    case '[':		/* oh dear */
	      read_set((unsigned char *)match,s-match+1,set,match_len);
	      for(e=0;e<256;e++) set[e]=!set[e];
	      goto match_set;
	    }
	  }

	  contains_percent_percent=0;

	  for(e=cnt;e<match_len;e++)
	  {
	    if(match[e]=='%')
	    {
	      if(match[e+1]=='%')
	      {
		contains_percent_percent=1;
		e++;
	      }else{
		break;
	      }
	    }
	  }
	   
	  end_str_end=match+e;

	  if(!contains_percent_percent)
	  {
	    s=my_memmem(end_str_start,
			end_str_end-end_str_start,
			input+eye,
			input_len-eye);
	    if(!s)
	    {
	      chars_matched[0]=eye;
	      return matches;
	    }
	    eye=s-input;
	    new_eye=eye+end_str_end-end_str_start;
	  }else{
	    for(;eye<input_len;eye++)
	    {
	      p=input+eye;
	      for(s=end_str_start;s<end_str_end;s++,p++)
	      {
		if(*s!=*p) break;
		if(*s=='%') s++;
	      }
	      if(s==end_str_end)
		break;
	    }
	    if(eye==input_len)
	    {
	      chars_matched[0]=eye;
	      return matches;
	    }
	    new_eye=p-input;
	  }

	  sval.type=T_STRING;
#ifdef __CHECKER__
	  sval.subtype=0;
#endif
	  sval.u.string=make_shared_binary_string(input+start,eye-start);

	  cnt=end_str_end-match-1;
	  eye=new_eye;
	  break;
	}

      case '[':
	cnt=read_set((unsigned char *)match,cnt+1,set,match_len);

      match_set:
	for(e=eye;eye<input_len && set[EXTRACT_UCHAR(input+eye)];eye++);
	sval.type=T_STRING;
#ifdef __CHECKER__
	sval.subtype=0;
#endif
	sval.u.string=make_shared_binary_string(input+e,eye-e);
	break;

      case 'n':
	sval.type=T_INT;
	sval.subtype=NUMBER_NUMBER;
	sval.u.integer=eye;
	break;
    
      default:
	error("Unknown sscanf token %%%c\n",match[cnt]);
      }
      break;
    }
    matches++;

    if(no_assign)
    {
      free_svalue(&sval);
    }else{
      check_stack(1);
      *sp++=sval;
#ifdef DEBUG
      sval.type=99;
#endif
    }
  }
  chars_matched[0]=eye;
  success[0]=1;
  return matches;
}

void o_sscanf(INT32 args)
{
#ifdef DEBUG
  extern int t_flag;
#endif
  INT32 e,i;
  int x;
  long matched_chars;
  struct svalue *save_sp=sp;

  if(sp[-args].type != T_STRING)
    error("Bad argument 1 to sscanf().\n");

  if(sp[1-args].type != T_STRING)
    error("Bad argument 1 to sscanf().\n");

  i=really_low_sscanf(sp[-args].u.string->str,
		      sp[-args].u.string->len,
		      sp[1-args].u.string->str,
		      sp[1-args].u.string->len,
		      &matched_chars,
		      &x);

  if(sp-save_sp > args/2-1)
    error("Too few arguments for sscanf format.\n");

  for(x=0;x<sp-save_sp;x++)
    assign_lvalue(save_sp-args+2+x*2,save_sp+x);
  pop_n_elems(sp-save_sp +args);

#ifdef DEBUG
  if(t_flag >2)
  {
    int nonblock;
    if((nonblock=query_nonblocking(2)))
      set_nonblocking(2,0);
    
    fprintf(stderr,"-    Matches: %ld\n",(long)i);
    if(nonblock)
      set_nonblocking(2,1);
  }
#endif
  push_int(i);
}

void f_sscanf(INT32 args)
{
#ifdef DEBUG
  extern int t_flag;
#endif
  INT32 e,i;
  int x;
  long matched_chars;
  struct svalue *save_sp=sp;
  struct array *a;

  check_all_args("array_sscanf",args,BIT_STRING, BIT_STRING,0);

  i=really_low_sscanf(sp[-args].u.string->str,
		      sp[-args].u.string->len,
		      sp[1-args].u.string->str,
		      sp[1-args].u.string->len,
		      &matched_chars,
		      &x);

  a=aggregate_array(sp-save_sp);
  pop_n_elems(args);
  push_array(a);
}