summaryrefslogtreecommitdiff
path: root/rts
diff options
context:
space:
mode:
authorBen Gamari <ben@smart-cactus.org>2015-11-26 12:02:33 +0100
committerBen Gamari <ben@smart-cactus.org>2015-11-26 14:48:48 +0100
commitc4308b468c7fe723b6f0a1009b233a55260de503 (patch)
treee1bd38e290aa9700d59b4393bba15e721f4a6f71 /rts
parentd2a2d5ebb08caedcd2a2a7a9e06ed6f4cbba96e7 (diff)
downloadhaskell-c4308b468c7fe723b6f0a1009b233a55260de503.tar.gz
rts/Pool: Add poolTryTake
Diffstat (limited to 'rts')
-rw-r--r--rts/Pool.c41
-rw-r--r--rts/Pool.h10
2 files changed, 39 insertions, 12 deletions
diff --git a/rts/Pool.c b/rts/Pool.c
index 6c238075ef..b3e3446647 100644
--- a/rts/Pool.c
+++ b/rts/Pool.c
@@ -111,19 +111,40 @@ nat poolGetDesiredSize(Pool *pool) {
return pool->desired_size;
}
+// Try taking a PoolEntry with an item from a pool,
+// returning NULL if no items are available.
+static PoolEntry *poolTryTake_(Pool *pool) {
+ PoolEntry *ent = NULL;
+ if (pool->available != NULL) {
+ ent = pool->available;
+ pool->available = ent->next;
+ } else if (pool->current_size < pool->max_size) {
+ ent = stgMallocBytes(sizeof(PoolEntry), "pool_take");
+ ent->flags = 0;
+ ent->thing = pool->alloc_fn();
+ pool->current_size++;
+ } else {
+ return NULL;
+ }
+
+ ent->next = pool->taken;
+ pool->taken = ent;
+ return ent;
+}
+
+void *poolTryTake(Pool *pool) {
+ ACQUIRE_LOCK(&pool->mutex);
+ PoolEntry *ent = poolTryTake_(pool);
+ RELEASE_LOCK(&pool->mutex);
+ return ent ? ent->thing : NULL;
+}
+
void *poolTake(Pool *pool) {
PoolEntry *ent = NULL;
ACQUIRE_LOCK(&pool->mutex);
while (ent == NULL) {
- if (pool->available != NULL) {
- ent = pool->available;
- pool->available = ent->next;
- } else if (pool->current_size < pool->max_size) {
- ent = stgMallocBytes(sizeof(PoolEntry), "pool_take");
- ent->flags = 0;
- ent->thing = pool->alloc_fn();
- pool->current_size++;
- } else {
+ ent = poolTryTake_(pool);
+ if (!ent) {
#ifdef THREADED_RTS
waitCondition(&pool->cond, &pool->mutex);
#else
@@ -132,8 +153,6 @@ void *poolTake(Pool *pool) {
}
}
- ent->next = pool->taken;
- pool->taken = ent;
RELEASE_LOCK(&pool->mutex);
return ent->thing;
}
diff --git a/rts/Pool.h b/rts/Pool.h
index d1aeab5661..dd00412d27 100644
--- a/rts/Pool.h
+++ b/rts/Pool.h
@@ -43,7 +43,15 @@ void poolSetDesiredSize(Pool *pool, nat size);
/* Get the desired size of a pool */
nat poolGetDesiredSize(Pool *pool);
-/* Grab an available thing from a pool */
+/* Try to grab an available thing from a pool, returning NULL if no things
+ * are available.
+ */
+void *poolTryTake(Pool *pool);
+
+/* Grab an available thing from a pool. This will block if no elements are
+ * available in the case of a threaded runtime or abort in a single-threaded
+ * environment.
+ */
void *poolTake(Pool *pool);
/* Release a thing back to the pool from which it was taken */