Pushing weighted ketama code.
[awesomized/libmemcached] / libmemcached / memcached.c
1 /*
2 Memcached library
3 */
4 #include "common.h"
5
6 memcached_st *memcached_create(memcached_st *ptr)
7 {
8 memcached_result_st *result_ptr;
9
10 if (ptr == NULL)
11 {
12 ptr= (memcached_st *)malloc(sizeof(memcached_st));
13
14 if (!ptr)
15 return NULL; /* MEMCACHED_MEMORY_ALLOCATION_FAILURE */
16
17 memset(ptr, 0, sizeof(memcached_st));
18 ptr->is_allocated= MEMCACHED_ALLOCATED;
19 }
20 else
21 {
22 memset(ptr, 0, sizeof(memcached_st));
23 }
24 result_ptr= memcached_result_create(ptr, &ptr->result);
25 WATCHPOINT_ASSERT(result_ptr);
26 ptr->poll_timeout= MEMCACHED_DEFAULT_TIMEOUT;
27 ptr->connect_timeout= MEMCACHED_DEFAULT_TIMEOUT;
28 ptr->retry_timeout= 0;
29 ptr->distribution= MEMCACHED_DISTRIBUTION_MODULA;
30
31 return ptr;
32 }
33
34 void memcached_free(memcached_st *ptr)
35 {
36 /* If we have anything open, lets close it now */
37 memcached_quit(ptr);
38 server_list_free(ptr, ptr->hosts);
39 memcached_result_free(&ptr->result);
40
41 if (ptr->on_cleanup)
42 ptr->on_cleanup(ptr);
43
44 if (ptr->continuum)
45 {
46 if (ptr->call_free)
47 ptr->call_free(ptr, ptr->continuum);
48 else
49 free(ptr->continuum);
50 }
51
52 if (ptr->is_allocated == MEMCACHED_ALLOCATED)
53 {
54 if (ptr->call_free)
55 ptr->call_free(ptr, ptr);
56 else
57 free(ptr);
58 }
59 else
60 ptr->is_allocated= MEMCACHED_USED;
61 }
62
63 /*
64 clone is the destination, while ptr is the structure to clone.
65 If ptr is NULL the call is the same as if a memcached_create() was
66 called.
67 */
68 memcached_st *memcached_clone(memcached_st *clone, memcached_st *ptr)
69 {
70 memcached_return rc= MEMCACHED_SUCCESS;
71 memcached_st *new_clone;
72
73 if (ptr == NULL)
74 return memcached_create(clone);
75
76 if (ptr->is_allocated == MEMCACHED_USED)
77 {
78 WATCHPOINT_ASSERT(0);
79 return NULL;
80 }
81
82 new_clone= memcached_create(clone);
83
84 if (new_clone == NULL)
85 return NULL;
86
87 if (ptr->hosts)
88 rc= memcached_server_push(new_clone, ptr->hosts);
89
90 if (rc != MEMCACHED_SUCCESS)
91 {
92 memcached_free(new_clone);
93
94 return NULL;
95 }
96
97
98 new_clone->flags= ptr->flags;
99 new_clone->send_size= ptr->send_size;
100 new_clone->recv_size= ptr->recv_size;
101 new_clone->poll_timeout= ptr->poll_timeout;
102 new_clone->connect_timeout= ptr->connect_timeout;
103 new_clone->retry_timeout= ptr->retry_timeout;
104 new_clone->distribution= ptr->distribution;
105 new_clone->hash= ptr->hash;
106 new_clone->hash_continuum= ptr->hash_continuum;
107 new_clone->user_data= ptr->user_data;
108
109 new_clone->on_clone= ptr->on_clone;
110 new_clone->on_cleanup= ptr->on_cleanup;
111 new_clone->call_free= ptr->call_free;
112 new_clone->call_malloc= ptr->call_malloc;
113 new_clone->call_realloc= ptr->call_realloc;
114 new_clone->get_key_failure= ptr->get_key_failure;
115 new_clone->delete_trigger= ptr->delete_trigger;
116
117 if (ptr->prefix_key[0] != 0)
118 {
119 strcpy(new_clone->prefix_key, ptr->prefix_key);
120 new_clone->prefix_key_length= ptr->prefix_key_length;
121 }
122
123 rc= run_distribution(new_clone);
124 if (rc != MEMCACHED_SUCCESS)
125 {
126 memcached_free(new_clone);
127
128 return NULL;
129 }
130
131 if (ptr->on_clone)
132 ptr->on_clone(ptr, new_clone);
133
134 return new_clone;
135 }