1 /* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
5 * Copyright (C) 2011 Data Differential, http://datadifferential.com/
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions are
12 * * Redistributions of source code must retain the above copyright
13 * notice, this list of conditions and the following disclaimer.
15 * * Redistributions in binary form must reproduce the above
16 * copyright notice, this list of conditions and the following disclaimer
17 * in the documentation and/or other materials provided with the
20 * * The names of its contributors may not be used to endorse or
21 * promote products derived from this software without specific prior
24 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
25 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
26 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
27 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
28 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
29 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
30 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
31 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
32 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
33 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
34 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
38 #include <libmemcached/common.h>
39 #include <libmemcached/virtual_bucket.h>
46 struct memcached_virtual_bucket_t
{
50 struct bucket_t buckets
[];
53 memcached_return_t
memcached_virtual_bucket_create(memcached_st
*self
,
54 const uint32_t *host_map
,
55 const uint32_t *forward_map
,
56 const uint32_t buckets
,
57 const uint32_t replicas
)
59 if (! self
|| ! host_map
|| ! buckets
)
60 return MEMCACHED_INVALID_ARGUMENTS
;
62 memcached_virtual_bucket_free(self
);
64 struct memcached_virtual_bucket_t
*virtual_bucket
= (struct memcached_virtual_bucket_t
*)malloc(sizeof(struct memcached_virtual_bucket_t
) + sizeof(struct bucket_t
) *buckets
);
67 return MEMCACHED_MEMORY_ALLOCATION_FAILURE
;
69 virtual_bucket
->size
= buckets
;
70 virtual_bucket
->replicas
= replicas
;
71 self
->virtual_bucket
= virtual_bucket
;
73 for (uint32_t x
=0; x
< buckets
; x
++)
75 virtual_bucket
->buckets
[x
].master
= host_map
[x
];
78 virtual_bucket
->buckets
[x
].forward
= forward_map
[x
];
82 virtual_bucket
->buckets
[x
].forward
= 0;
87 return MEMCACHED_SUCCESS
;
90 void memcached_virtual_bucket_free(memcached_st
*self
)
95 if (! self
->virtual_bucket
)
98 free(self
->virtual_bucket
);
99 self
->virtual_bucket
= NULL
;
102 uint32_t memcached_virtual_bucket_get(const memcached_st
*self
, uint32_t digest
)
107 if (! self
->virtual_bucket
)
110 if (self
->virtual_bucket
)
112 uint32_t result
= (uint32_t) (digest
& (self
->virtual_bucket
->size
-1));
113 return self
->virtual_bucket
->buckets
[result
].master
;
116 return (uint32_t) (digest
& (self
->number_of_hosts
-1));