1
  
2
  
3
  
4
  
5
  
6
  
7
  
8
  
9
  
10
  
11
  
12
  
13
  
14
  
15
  
16
  
17
  
18
  
19
  
20
  
21
  
22
  
23
  
24
  
25
  
26
  
27
  
28
  
29
  
30
  
31
  
32
  
33
  
34
  
35
  
36
  
37
  
38
  
39
  
40
  
41
  
42
  
43
  
44
  
45
  
46
  
47
  
48
  
49
  
50
  
51
  
52
  
53
  
54
  
55
  
56
  
57
  
58
  
59
  
60
  
61
  
62
  
63
  
64
  
65
  
66
  
67
  
68
  
69
  
70
  
71
  
72
  
73
  
74
  
75
  
76
  
77
  
78
  
79
  
80
  
81
  
82
  
83
  
84
  
85
  
86
  
87
  
88
  
89
  
90
  
91
  
92
  
93
  
94
  
95
  
96
  
97
  
98
  
99
  
100
  
101
  
102
  
103
  
104
  
105
  
106
  
107
  
108
  
109
  
110
  
111
  
112
  
113
  
114
  
115
  
116
  
117
  
118
  
119
  
120
  
121
  
#include "global.h" 
#include "bignum.h" 
#include "object.h" 
#include "interpret.h" 
 
#include <shuffler.h> 
 
/* $Id: a_source_system_memory.c,v 1.4 2002/05/29 09:33:13 per Exp $ */ 
 
/* Source: System.Memory 
 * Argument: An initialized instance of the System.Memory class 
 */ 
static struct program *shm_program; 
 
struct sm_source 
{ 
  struct source s; 
 
  struct object *obj; 
  struct { 
    unsigned char *data; 
    size_t len; 
  } *mem; 
 
  int offset, len; 
}; 
 
static struct data get_data( struct source *_s, int len ) 
{ 
  struct sm_source *s = (struct sm_source *)s; 
  struct data res; 
   
  res.do_free = 0; 
  res.off = 0; 
  res.data = s->mem->data + s->offset; 
   
  if( len > s->len ) 
  { 
    len = s->len; 
    s->s.eof = 1; /* next read will be done from the next source */ 
  } 
 
  res.len = len; 
 
  s->len -= len; 
  s->offset += len; 
 
  return res; 
} 
 
static void free_source( struct source *_s ) 
{ 
  free_object(((struct sm_source *)_s)->obj); 
} 
 
struct source *source_system_memory_make( struct svalue *s, 
                                          INT64 start, INT64 len ) 
{ 
  struct sm_source *res; 
 
  if( s->type != PIKE_T_OBJECT ) 
    return 0; 
 
   
  res = malloc( sizeof( struct sm_source ) ); 
  MEMSET( res, 0, sizeof( struct sm_source ) ); 
 
  if( !(res->mem = (void*)get_storage( s->u.object, shm_program ) ) ) 
  { 
    free(res); 
    return 0; 
  } 
 
  if( !res->mem->data || !res->mem->len ) 
  { 
    free(res); 
    return 0; 
  } 
   
  res->s.free_source = free_source; 
  res->s.get_data = get_data; 
  res->obj = s->u.object; 
  res->obj->refs++; 
  res->offset = start; 
 
  if( len != -1 ) 
    if( len > res->mem->len-start ) 
    { 
      res->obj->refs--; 
      free(res); 
      return 0; 
    } 
    else 
      res->len = len; 
  else 
    res->len = len; 
 
  if( res->len <= 0 ) 
  { 
    res->obj->refs--; 
    free(res); 
    return 0; 
  } 
  return (struct source *)res; 
} 
 
void source_system_memory_exit( ) 
{ 
  free_program( shm_program ); 
} 
 
void source_system_memory_init( ) 
{ 
  push_text("System.Memory"); push_int(0); 
  SAFE_APPLY_MASTER("resolv",2); 
  shm_program = program_from_svalue(Pike_sp-1); 
  shm_program->refs++; 
  pop_stack( ); 
}