blob: a2628cc1b5c9d0c1b9f9555b0b2292d0ed9a8db8 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 <nuttx/config.h>
#include <assert.h>
#include <stdint.h>
#include <string.h>
#include <fcntl.h>
#include <stdlib.h>
#include <stdio.h>
#include "nimble/nimble_npl.h"
static struct ble_npl_eventq dflt_evq =
{
.mq = (mqd_t)-1
};
int qnum = 0;
struct ble_npl_eventq *
ble_npl_eventq_dflt_get(void)
{
return &dflt_evq;
}
void
ble_npl_eventq_init(struct ble_npl_eventq *evq)
{
char path[PATH_MAX];
struct mq_attr attr;
attr.mq_maxmsg = 16;
attr.mq_msgsize = sizeof(struct ble_npl_event*);
attr.mq_flags = 0;
qnum++;
sprintf(path, "/nimbleq%i", qnum);
evq->mq = mq_open(path, O_CREAT | O_RDWR, 0, &attr);
DEBUGASSERT((intptr_t)evq->mq >= 0);
}
bool
ble_npl_eventq_is_empty(struct ble_npl_eventq *evq)
{
struct mq_attr attr;
mq_getattr(evq->mq, &attr);
return (attr.mq_curmsgs == 0);
}
int
ble_npl_eventq_inited(const struct ble_npl_eventq *evq)
{
return (evq->mq != ((mqd_t)-1));
}
void
ble_npl_eventq_put(struct ble_npl_eventq *evq, struct ble_npl_event *ev)
{
int ret;
if (ev->ev_queued)
{
return;
}
ev->ev_queued = 1;
ret = mq_send(evq->mq, (const char*)&ev, sizeof(ev), 0);
DEBUGASSERT(ret == 0);
UNUSED(ret);
}
struct ble_npl_event *
ble_npl_eventq_get(struct ble_npl_eventq *evq,
ble_npl_time_t tmo)
{
struct ble_npl_event *ev = NULL;
if (tmo == BLE_NPL_TIME_FOREVER)
{
do
{
mq_receive(evq->mq, (char*)&ev, sizeof(ev), NULL);
} while (ev && ev->ev_queued == 0);
}
else
{
struct timespec now, t;
clock_gettime(CLOCK_REALTIME, &now);
t.tv_sec += tmo / USEC_PER_SEC;
t.tv_nsec = (tmo - t.tv_sec) * NSEC_PER_MSEC;
clock_timespec_add(&now, &t, &t);
do
{
mq_timedreceive(evq->mq, (char*)&ev, sizeof(ev), NULL, &t);
} while (ev && ev->ev_queued == 0);
}
if (ev) {
ev->ev_queued = 0;
}
return ev;
}
void
ble_npl_eventq_run(struct ble_npl_eventq *evq)
{
struct ble_npl_event *ev;
ev = ble_npl_eventq_get(evq, BLE_NPL_TIME_FOREVER);
ble_npl_event_run(ev);
}
// ========================================================================
// Event Implementation
// ========================================================================
void
ble_npl_event_init(struct ble_npl_event *ev, ble_npl_event_fn *fn,
void *arg)
{
memset(ev, 0, sizeof(*ev));
ev->ev_cb = fn;
ev->ev_arg = arg;
}
bool
ble_npl_event_is_queued(struct ble_npl_event *ev)
{
return ev->ev_queued;
}
void *
ble_npl_event_get_arg(struct ble_npl_event *ev)
{
return ev->ev_arg;
}
void
ble_npl_event_set_arg(struct ble_npl_event *ev, void *arg)
{
ev->ev_arg = arg;
}
void
ble_npl_event_run(struct ble_npl_event *ev)
{
assert(ev->ev_cb != NULL);
ev->ev_cb(ev);
}
void
ble_npl_eventq_remove(struct ble_npl_eventq *evq, struct ble_npl_event *ev)
{
/* we simply mark the event as unqueued. we will ignore these elements
* when receiving from the queue */
ev->ev_queued = 0;
}