Dependencies:   MMA7660 LM75B

simple-mbed-cloud-client/mbed-cloud-client/update-client-hub/modules/atomic-queue/source/atomic-queue.c

Committer:
MACRUM
Date:
2018-06-30
Revision:
0:119624335925

File content as of revision 0:119624335925:

// ----------------------------------------------------------------------------
// Copyright 2015-2017 ARM Ltd.
//
// SPDX-License-Identifier: Apache-2.0
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// ----------------------------------------------------------------------------

#include "atomic-queue/atomic-queue.h"
#include "atomic.h"

#include <stddef.h>
#include <stdint.h>
#include <stdio.h>

#define CORE_UTIL_ASSERT_MSG(test, msg)


int aq_push_tail(struct atomic_queue * q, struct atomic_queue_element * e)
{
    CORE_UTIL_ASSERT_MSG(q != NULL, "null queue used");
    if (q == NULL) {
        return ATOMIC_QUEUE_NULL_QUEUE;
    }

/* duplicate element check using lock */
#ifdef ATOMIC_QUEUE_CONFIG_ELEMENT_LOCK
    uintptr_t lock;
    // Check/obtain a lock on the element.
    do {
        lock = e->lock;
        if (lock) {
            return ATOMIC_QUEUE_DUPLICATE_ELEMENT;
        }
    } while (!aq_atomic_cas_uintptr(&e->lock, lock, 1));
#endif

    do {
/* duplicate element check by searching */
#ifndef ATOMIC_QUEUE_CONFIG_ELEMENT_LOCK
        // Make sure the new element does not exist in the current queue
        struct atomic_queue_element* te = q->tail;
        while (te != NULL)
        {
            CORE_UTIL_ASSERT_MSG(te != e, "duplicate queue element");
            if (te == e) {
                return ATOMIC_QUEUE_DUPLICATE_ELEMENT;
            }
            te = te->next;
        }
#endif
        e->next = q->tail;
    } while (!aq_atomic_cas_uintptr((uintptr_t *)&q->tail, (uintptr_t)e->next, (uintptr_t)e));

    return ATOMIC_QUEUE_SUCCESS;
}

struct atomic_queue_element * aq_pop_head(struct atomic_queue * q)
{
    CORE_UTIL_ASSERT_MSG(q != NULL, "null queue used");
    if (q == NULL) {
        return NULL;
    }
    struct atomic_queue_element * current;
    int fail = AQ_ATOMIC_CAS_DEREF_VALUE;
    while (fail != AQ_ATOMIC_CAS_DEREF_SUCCESS) {
        // Set the element reference pointer to the tail pointer
        struct atomic_queue_element * volatile * px = &q->tail;
        if (*px == NULL) {
            return NULL;
        }
        fail = AQ_ATOMIC_CAS_DEREF_VALUE;
        while (fail == AQ_ATOMIC_CAS_DEREF_VALUE) {
            fail = aq_atomic_cas_deref_uintptr((uintptr_t * volatile *)px,
                            (uintptr_t **)&current,
                            (uintptr_t) NULL,
                            NULL,
                            offsetof(struct atomic_queue_element, next));
            if (fail == AQ_ATOMIC_CAS_DEREF_VALUE) {
                if (current->next == q->tail) {
                    return NULL;
                }
                px = &current->next;
            }
        }
    }

#ifdef ATOMIC_QUEUE_CONFIG_ELEMENT_LOCK
    // Release element lock
    current->lock = 0;
#endif

    return current;
}


int aq_empty(struct atomic_queue * q)
{
    return q->tail == NULL;
}

unsigned aq_count(struct atomic_queue *q)
{
    unsigned x;
    struct atomic_queue_element * volatile e;
    if (aq_empty(q)) {
        return 0;
    }
    e = q->tail;
    for (x = 1; e->next != NULL; x++, e = e->next) {
        if (e->next == q->tail){
            return (unsigned)-1;
        }
    }
    return x;
}

void aq_initialize_element(struct atomic_queue_element* e)
{
#ifdef ATOMIC_QUEUE_CONFIG_ELEMENT_LOCK
    e->lock = 0;
#endif
}