-
Notifications
You must be signed in to change notification settings - Fork 0
/
apc_cache.h
372 lines (323 loc) · 15.1 KB
/
apc_cache.h
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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
/*
+----------------------------------------------------------------------+
| APC |
+----------------------------------------------------------------------+
| Copyright (c) 2006-2011 The PHP Group |
+----------------------------------------------------------------------+
| This source file is subject to version 3.01 of the PHP license, |
| that is bundled with this package in the file LICENSE, and is |
| available through the world-wide-web at the following url: |
| http://www.php.net/license/3_01.txt. |
| If you did not receive a copy of the PHP license and are unable to |
| obtain it through the world-wide-web, please send a note to |
| [email protected] so we can mail you a copy immediately. |
+----------------------------------------------------------------------+
| Authors: Daniel Cowgill <[email protected]> |
| Rasmus Lerdorf <[email protected]> |
+----------------------------------------------------------------------+
This software was contributed to PHP by Community Connect Inc. in 2002
and revised in 2005 by Yahoo! Inc. to add support for PHP 5.1.
Future revisions and derivatives of this source code must acknowledge
Community Connect Inc. as the original contributor of this module by
leaving this note intact in the source code.
All other licensing and usage conditions are those of the PHP Group.
*/
/* $Id: apc_cache.h 326865 2012-07-29 14:19:27Z ab $ */
#ifndef APC_CACHE_H
#define APC_CACHE_H
/*
* This module defines the shared memory file cache. Basically all of the
* logic for storing and retrieving cache entries lives here.
*/
#include "apc.h"
#include "apc_compile.h"
#include "apc_lock.h"
#include "apc_pool.h"
#include "apc_main.h"
#include "TSRM.h"
#define APC_CACHE_ENTRY_FILE 1
#define APC_CACHE_ENTRY_USER 2
#define APC_CACHE_KEY_FILE 1
#define APC_CACHE_KEY_USER 2
#define APC_CACHE_KEY_FPFILE 3
#ifdef PHP_WIN32
typedef unsigned __int64 apc_ino_t;
typedef unsigned __int64 apc_dev_t;
#else
typedef ino_t apc_ino_t;
typedef dev_t apc_dev_t;
#endif
/* {{{ cache locking macros */
#define CACHE_LOCK(cache) { LOCK(cache->header->lock); cache->has_lock = 1; }
#define CACHE_UNLOCK(cache) { UNLOCK(cache->header->lock); cache->has_lock = 0; }
#define CACHE_SAFE_LOCK(cache) { if ((++cache->has_lock) == 1) LOCK(cache->header->lock); }
#define CACHE_SAFE_UNLOCK(cache) { if ((--cache->has_lock) == 0) UNLOCK(cache->header->lock); }
#if (RDLOCK_AVAILABLE == 1) && defined(HAVE_ATOMIC_OPERATIONS)
#define USE_READ_LOCKS 1
#define CACHE_RDLOCK(cache) { RDLOCK(cache->header->lock); cache->has_lock = 0; }
#define CACHE_RDUNLOCK(cache) { RDUNLOCK(cache->header->lock); cache->has_lock = 0; }
#define CACHE_SAFE_INC(cache, obj) { ATOMIC_INC(obj); }
#define CACHE_SAFE_DEC(cache, obj) { ATOMIC_DEC(obj); }
#else
#define USE_READ_LOCKS 0
#define CACHE_RDLOCK(cache) { LOCK(cache->header->lock); cache->has_lock = 1; }
#define CACHE_RDUNLOCK(cache) { UNLOCK(cache->header->lock); cache->has_lock = 0; }
#define CACHE_SAFE_INC(cache, obj) { CACHE_SAFE_LOCK(cache); obj++; CACHE_SAFE_UNLOCK(cache);}
#define CACHE_SAFE_DEC(cache, obj) { CACHE_SAFE_LOCK(cache); obj--; CACHE_SAFE_UNLOCK(cache);}
#endif
#define CACHE_FAST_INC(cache, obj) { obj++; }
#define CACHE_FAST_DEC(cache, obj) { obj--; }
/* }}} */
/* {{{ struct definition: apc_cache_key_t */
#define T apc_cache_t*
typedef struct apc_cache_t apc_cache_t; /* opaque cache type */
typedef union _apc_cache_key_data_t {
struct {
apc_dev_t device; /* the filesystem device */
apc_ino_t inode; /* the filesystem inode */
} file;
struct {
const char *identifier;
int identifier_len;
} user;
struct {
const char *fullpath;
int fullpath_len;
} fpfile;
} apc_cache_key_data_t;
typedef struct apc_cache_key_t apc_cache_key_t;
struct apc_cache_key_t {
apc_cache_key_data_t data;
unsigned long h; /* pre-computed hash value */
time_t mtime; /* the mtime of this cached entry */
unsigned char type;
unsigned char md5[16]; /* md5 hash of the source file */
};
typedef struct apc_keyid_t apc_keyid_t;
struct apc_keyid_t {
unsigned int h;
unsigned int keylen;
time_t mtime;
#ifdef ZTS
THREAD_T tid;
#else
pid_t pid;
#endif
};
/* }}} */
/* {{{ struct definition: apc_cache_entry_t */
typedef union _apc_cache_entry_value_t {
struct {
char *filename; /* absolute path to source file */
zend_op_array* op_array; /* op_array allocated in shared memory */
apc_function_t* functions; /* array of apc_function_t's */
apc_class_t* classes; /* array of apc_class_t's */
long halt_offset; /* value of __COMPILER_HALT_OFFSET__ for the file */
zend_uint *exec_refcount; /* refcount member of zend_op_array refreshed before execution */
} file;
struct {
char *info;
int info_len;
zval *val;
unsigned int ttl;
} user;
} apc_cache_entry_value_t;
typedef struct apc_cache_entry_t apc_cache_entry_t;
struct apc_cache_entry_t {
apc_cache_entry_value_t data;
unsigned char type;
int ref_count;
size_t mem_size;
apc_pool *pool;
};
/* }}} */
/*
* apc_cache_create creates the shared memory compiler cache. This function
* should be called just once (ideally in the web server parent process, e.g.
* in apache), otherwise you will end up with multiple caches (which won't
* necessarily break anything). Returns a pointer to the cache object.
*
* size_hint is a "hint" at the total number of source files that will be
* cached. It determines the physical size of the hash table. Passing 0 for
* this argument will use a reasonable default value.
*
* gc_ttl is the maximum time a cache entry may speed on the garbage
* collection list. This is basically a work around for the inherent
* unreliability of our reference counting mechanism (see apc_cache_release).
*
* ttl is the maximum time a cache entry can idle in a slot in case the slot
* is needed. This helps in cleaning up the cache and ensuring that entries
* hit frequently stay cached and ones not hit very often eventually disappear.
*/
extern T apc_cache_create(int size_hint, int gc_ttl, int ttl TSRMLS_DC);
/*
* apc_cache_destroy releases any OS resources associated with a cache object.
* Under apache, this function can be safely called by the child processes
* when they exit.
*/
extern void apc_cache_destroy(T cache TSRMLS_DC);
/*
* apc_cache_clear empties a cache. This can safely be called at any time,
* even while other server processes are executing cached source files.
*/
extern void apc_cache_clear(T cache TSRMLS_DC);
/*
* apc_cache_insert adds an entry to the cache, using a filename as a key.
* Internally, the filename is translated to a canonical representation, so
* that relative and absolute filenames will map to a single key. Returns
* non-zero if the file was successfully inserted, 0 otherwise. If 0 is
* returned, the caller must free the cache entry by calling
* apc_cache_free_entry (see below).
*
* key is the value created by apc_cache_make_file_key for file keys.
*
* value is a cache entry returned by apc_cache_make_entry (see below).
*/
extern int apc_cache_insert(T cache, apc_cache_key_t key,
apc_cache_entry_t* value, apc_context_t* ctxt, time_t t TSRMLS_DC);
extern int apc_cache_user_insert(T cache, apc_cache_key_t key,
apc_cache_entry_t* value, apc_context_t* ctxt, time_t t, int exclusive TSRMLS_DC);
extern int *apc_cache_insert_mult(apc_cache_t* cache, apc_cache_key_t* keys,
apc_cache_entry_t** values, apc_context_t *ctxt, time_t t, int num_entries TSRMLS_DC);
/*
* apc_cache_find searches for a cache entry by filename, and returns a
* pointer to the entry if found, NULL otherwise.
*
* key is a value created by apc_cache_make_file_key for file keys.
*/
extern apc_cache_entry_t* apc_cache_find(T cache, apc_cache_key_t key, time_t t TSRMLS_DC);
/*
* apc_cache_user_find searches for a cache entry by its hashed identifier,
* and returns a pointer to the entry if found, NULL otherwise.
*
*/
extern apc_cache_entry_t* apc_cache_user_find(T cache, char* strkey, int keylen, time_t t TSRMLS_DC);
/*
* apc_cache_user_exists searches for a cache entry by its hashed identifier,
* and returns a pointer to the entry if found, NULL otherwise. This is a
* quick non-locking version of apc_cache_user_find that does not modify the
* shared memory segment in any way.
*
*/
extern apc_cache_entry_t* apc_cache_user_exists(T cache, char* strkey, int keylen, time_t t TSRMLS_DC);
/*
* apc_cache_delete and apc_cache_user_delete finds an entry in the cache and deletes it.
*/
extern int apc_cache_delete(apc_cache_t* cache, char *filename, int filename_len TSRMLS_DC);
extern int apc_cache_user_delete(apc_cache_t* cache, char *strkey, int keylen TSRMLS_DC);
/* apc_cach_fetch_zval takes a zval in the cache and reconstructs a runtime
* zval from it.
*
*/
zval* apc_cache_fetch_zval(zval* dst, const zval* src, apc_context_t* ctxt TSRMLS_DC);
/*
* apc_cache_release decrements the reference count associated with a cache
* entry. Calling apc_cache_find automatically increments the reference count,
* and this function must be called post-execution to return the count to its
* original value. Failing to do so will prevent the entry from being
* garbage-collected.
*
* entry is the cache entry whose ref count you want to decrement.
*/
extern void apc_cache_release(T cache, apc_cache_entry_t* entry TSRMLS_DC);
/*
* apc_cache_make_file_key creates a key object given a relative or absolute
* filename and an optional list of auxillary paths to search. include_path is
* searched if the filename cannot be found relative to the current working
* directory.
*
* key points to caller-allocated storage (must not be null).
*
* filename is the path to the source file.
*
* include_path is a colon-separated list of directories to search.
*
* and finally we pass in the current request time so we can avoid
* caching files with a current mtime which tends to indicate that
* they are still being written to.
*/
extern int apc_cache_make_file_key(apc_cache_key_t* key,
const char* filename,
const char* include_path,
time_t t
TSRMLS_DC);
/*
* apc_cache_make_file_entry creates an apc_cache_entry_t object given a filename
* and the compilation results returned by the PHP compiler.
*/
extern apc_cache_entry_t* apc_cache_make_file_entry(const char* filename,
zend_op_array* op_array,
apc_function_t* functions,
apc_class_t* classes,
apc_context_t* ctxt
TSRMLS_DC);
zend_bool apc_compile_cache_entry(apc_cache_key_t *key, zend_file_handle* h, int type, time_t t, zend_op_array** op_array_pp, apc_cache_entry_t** cache_entry_pp TSRMLS_DC);
/*
* apc_cache_make_user_entry creates an apc_cache_entry_t object given an info string
* and the zval to be stored.
*/
extern apc_cache_entry_t* apc_cache_make_user_entry(const char* info, int info_len, const zval *val, apc_context_t* ctxt, const unsigned int ttl TSRMLS_DC);
extern int apc_cache_make_user_key(apc_cache_key_t* key, char* identifier, int identifier_len, const time_t t);
/* {{{ struct definition: slot_t */
typedef struct slot_t slot_t;
struct slot_t {
apc_cache_key_t key; /* slot key */
apc_cache_entry_t* value; /* slot value */
slot_t* next; /* next slot in linked list */
unsigned long num_hits; /* number of hits to this bucket */
time_t creation_time; /* time slot was initialized */
time_t deletion_time; /* time slot was removed from cache */
time_t access_time; /* time slot was last accessed */
};
/* }}} */
/* {{{ struct definition: cache_header_t
Any values that must be shared among processes should go in here. */
typedef struct cache_header_t cache_header_t;
struct cache_header_t {
apc_lck_t lock; /* read/write lock (exclusive blocking cache lock) */
apc_lck_t wrlock; /* write lock (non-blocking used to prevent cache slams) */
unsigned long num_hits; /* total successful hits in cache */
unsigned long num_misses; /* total unsuccessful hits in cache */
unsigned long num_inserts; /* total successful inserts in cache */
unsigned long expunges; /* total number of expunges */
slot_t* deleted_list; /* linked list of to-be-deleted slots */
time_t start_time; /* time the above counters were reset */
zend_bool busy; /* Flag to tell clients when we are busy cleaning the cache */
int num_entries; /* Statistic on the number of entries */
size_t mem_size; /* Statistic on the memory size used by this cache */
apc_keyid_t lastkey; /* the key that is being inserted (user cache) */
};
/* }}} */
typedef void (*apc_expunge_cb_t)(T cache, size_t n TSRMLS_DC);
/* {{{ struct definition: apc_cache_t */
struct apc_cache_t {
void* shmaddr; /* process (local) address of shared cache */
cache_header_t* header; /* cache header (stored in SHM) */
slot_t** slots; /* array of cache slots (stored in SHM) */
int num_slots; /* number of slots in cache */
int gc_ttl; /* maximum time on GC list for a slot */
int ttl; /* if slot is needed and entry's access time is older than this ttl, remove it */
apc_expunge_cb_t expunge_cb; /* cache specific expunge callback to free up sma memory */
uint has_lock; /* flag for possible recursive locks within the same process */
};
/* }}} */
extern zval* apc_cache_info(T cache, zend_bool limited TSRMLS_DC);
extern void apc_cache_unlock(apc_cache_t* cache TSRMLS_DC);
extern zend_bool apc_cache_busy(apc_cache_t* cache);
extern zend_bool apc_cache_write_lock(apc_cache_t* cache TSRMLS_DC);
extern void apc_cache_write_unlock(apc_cache_t* cache TSRMLS_DC);
extern zend_bool apc_cache_is_last_key(apc_cache_t* cache, apc_cache_key_t* key, time_t t TSRMLS_DC);
/* used by apc_rfc1867 to update data in-place - not to be used elsewhere */
typedef int (*apc_cache_updater_t)(apc_cache_t*, apc_cache_entry_t*, void* data);
extern int _apc_cache_user_update(apc_cache_t* cache, char *strkey, int keylen,
apc_cache_updater_t updater, void* data TSRMLS_DC);
#undef T
#endif
/*
* Local variables:
* tab-width: 4
* c-basic-offset: 4
* End:
* vim>600: expandtab sw=4 ts=4 sts=4 fdm=marker
* vim<600: expandtab sw=4 ts=4 sts=4
*/