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>
45 struct memcached_virtual_bucket_t
{
49 struct bucket_t buckets
[];
52 memcached_return_t
memcached_virtual_bucket_create(memcached_st
*self
,
53 const uint32_t *host_map
,
54 const uint32_t *forward_map
,
55 const uint32_t buckets
,
56 const uint32_t replicas
)
58 if (self
== NULL
|| host_map
== NULL
|| buckets
== 0U)
60 return MEMCACHED_INVALID_ARGUMENTS
;
63 memcached_virtual_bucket_free(self
);
65 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 if (virtual_bucket
== NULL
)
69 return MEMCACHED_MEMORY_ALLOCATION_FAILURE
;
73 virtual_bucket
->size
= buckets
;
74 virtual_bucket
->replicas
= replicas
;
75 self
->virtual_bucket
= virtual_bucket
;
78 for (; x
< buckets
; x
++)
80 virtual_bucket
->buckets
[x
].master
= host_map
[x
];
83 virtual_bucket
->buckets
[x
].forward
= forward_map
[x
];
87 virtual_bucket
->buckets
[x
].forward
= 0;
91 return MEMCACHED_SUCCESS
;
94 void memcached_virtual_bucket_free(memcached_st
*self
)
98 if (self
->virtual_bucket
)
100 free(self
->virtual_bucket
);
101 self
->virtual_bucket
= NULL
;
106 uint32_t memcached_virtual_bucket_get(const memcached_st
*self
, uint32_t digest
)
110 if (self
->virtual_bucket
)
112 if (self
->virtual_bucket
)
114 uint32_t result
= (uint32_t) (digest
& (self
->virtual_bucket
->size
-1));
115 return self
->virtual_bucket
->buckets
[result
].master
;
118 return (uint32_t) (digest
& (self
->number_of_hosts
-1));