OpenCores
URL https://opencores.org/ocsvn/openrisc/openrisc/trunk

Subversion Repositories openrisc

[/] [openrisc/] [trunk/] [rtos/] [ecos-3.0/] [packages/] [net/] [lwip_tcpip/] [current/] [src/] [ecos/] [sys_arch.c] - Rev 868

Go to most recent revision | Compare with Previous | Blame | View Log

//==========================================================================
//
//      sys_arch.c
//
//      lwIP system architecture support.
//
//==========================================================================
//####ECOSGPLCOPYRIGHTBEGIN####
// -------------------------------------------
// This file is part of eCos, the Embedded Configurable Operating System.
// Copyright (C) 2008, 2009 Free Software Foundation
//
// eCos is free software; you can redistribute it and/or modify it under
// the terms of the GNU General Public License as published by the Free
// Software Foundation; either version 2 or (at your option) any later version.
//
// eCos is distributed in the hope that it will be useful, but WITHOUT ANY
// WARRANTY; without even the implied warranty of MERCHANTABILITY or
// FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
// for more details.
//
// You should have received a copy of the GNU General Public License along
// with eCos; if not, write to the Free Software Foundation, Inc.,
// 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
//
// As a special exception, if other files instantiate templates or use macros
// or inline functions from this file, or you compile this file and link it
// with other works to produce a work based on this file, this file does not
// by itself cause the resulting work to be covered by the GNU General Public
// License. However the source code for this file must still be made available
// in accordance with section (3) of the GNU General Public License.
//
// This exception does not invalidate any other reasons why a work based on
// this file might be covered by the GNU General Public License.
// -------------------------------------------
//####ECOSGPLCOPYRIGHTEND####
//==========================================================================
//#####DESCRIPTIONBEGIN####
//
// Author(s):    Simon Kallweit
// Contributors:
// Date:         2008-12-02
// Purpose:
// Description:  Provides the system architecture support for lwIP.
//
//####DESCRIPTIONEND####
//
//==========================================================================
 
#include <pkgconf/hal.h>
 
#include <cyg/hal/hal_arch.h>
#include <cyg/kernel/kapi.h>
 
#include <lwip.h>
 
#include "lwip/opt.h"
#include "arch/sys_arch.h"
#include "lwip/sys.h"
#include "lwip/def.h"
#include "lwip/stats.h"
#include "lwip/debug.h"
 
// Milliseconds per system tick
#define MS_PER_TICK ((u32_t) (CYGNUM_HAL_RTC_NUMERATOR / \
                     (CYGNUM_HAL_RTC_DENOMINATOR * 1000000LL)))
 
// Macros to convert between ticks and milliseconds
#define TICKS_TO_MS(_ticks_)    ((u16_t) ((_ticks_) * MS_PER_TICK + 1))
#define MS_TO_TICKS(_ms_)       ((cyg_tick_count_t) (((_ms_) + \
                                 (MS_PER_TICK - 1)) / MS_PER_TICK))
 
#if !NO_SYS
 
// Thread structure
struct lwip_thread {
    struct lwip_thread *next;  // Next thread in linked list
    struct sys_timeouts to;    // List of timeouts
    cyg_handle_t handle;       // Thread handle
    cyg_thread thread;         // Thread store
};
 
// A var memory pool is used for allocating semaphores, mboxes and threads
static char var_data[CYGNUM_LWIP_VARMEMPOOL_SIZE];
static cyg_mempool_var var_mempool;
static cyg_handle_t var_handle;
 
// Internal lwip thread stacks are statically allocated
#define TOTAL_STACKSIZE     (TCPIP_THREAD_STACKSIZE +       \
                             SLIPIF_THREAD_STACKSIZE +      \
                             PPP_THREAD_STACKSIZE +         \
                             ETH_THREAD_STACKSIZE)
 
static cyg_mutex_t stack_mutex;
static char stack_data[TOTAL_STACKSIZE];
static char *stack_pos = stack_data;
 
// Timeout for threads which were not created by sys_thread_new()
static struct sys_timeouts to;
 
// List of threads
static struct lwip_thread *threads;
 
//
// Is called to initialize the sys_arch layer.
//
void
sys_init(void)
{
    cyg_mempool_var_create(
	    var_data,
	    sizeof(var_data),
	    &var_handle,
	    &var_mempool
	);
 
	threads = NULL;
	to.next = NULL;
 
	cyg_mutex_init(&stack_mutex);
}
 
//
// Creates and returns a new semaphore. The "count" argument specifies the
// initial state of the semaphore.
//
sys_sem_t
sys_sem_new(u8_t count)
{
    sys_sem_t sem;
 
    // Allocate semaphore
    sem = (cyg_sem_t *) cyg_mempool_var_try_alloc(var_handle, sizeof(cyg_sem_t));
    if (!sem)
        return SYS_SEM_NULL;
    cyg_semaphore_init(sem, count);
 
#if SYS_STATS
    lwip_stats.sys.sem.used++;
    if (lwip_stats.sys.sem.used > lwip_stats.sys.sem.max)
        lwip_stats.sys.sem.max = lwip_stats.sys.sem.used;
#endif
 
    return sem;
}
 
//
// Deallocates a semaphore.
//
void
sys_sem_free(sys_sem_t sem)
{
    if (!sem)
        return;
 
    cyg_semaphore_destroy(sem);
    cyg_mempool_var_free(var_handle, (void *) sem);
 
#if SYS_STATS
    lwip_stats.sys.sem.used--;
#endif
}
 
//
// Signals a semaphore.
//
void
sys_sem_signal(sys_sem_t sem)
{
    cyg_semaphore_post(sem);
}
 
//
// Blocks the thread while waiting for the semaphore to be signaled. If the
// "timeout" argument is non-zero, the thread should only be blocked for the
// specified time (measured in milliseconds). If the "timeout" argument is
// zero, the thread should be blocked until the semaphore is signalled.
//
// If the timeout argument is non-zero, the return value is the number of
// milliseconds spent waiting for the semaphore to be signaled. If the
// semaphore wasn't signaled within the specified time, the return value is
// SYS_ARCH_TIMEOUT. If the thread didn't have to wait for the semaphore
// (i.e., it was already signaled), the function may return zero.
//
// Notice that lwIP implements a function with a similar name, sys_sem_wait(),
// that uses the sys_arch_sem_wait() function.
//
u32_t
sys_arch_sem_wait(sys_sem_t sem, u32_t timeout)
{
    if (timeout) {
        cyg_tick_count_t start_time = cyg_current_time();
 
        // Wait for semaphore with timeout
        if (!cyg_semaphore_timed_wait(sem, start_time + MS_TO_TICKS(timeout)))
            return SYS_ARCH_TIMEOUT;
        // Return elapsed time
        return TICKS_TO_MS(cyg_current_time() - start_time);
    } else {
        // Wait for semaphore indefinitely
        cyg_semaphore_wait(sem);
        return 0;
    }
}
 
//
// Creates an empty mailbox for maximum "size" elements. Elements stored in
// mailboxes are pointers. You have to define macros "_MBOX_SIZE" in your
// lwipopts.h, or ignore this parameter in your implementation and use a
// default size.
//
sys_mbox_t
sys_mbox_new(int size)
{
	cyg_mbox *mbox;
	cyg_handle_t handle;
 
    LWIP_UNUSED_ARG(size);
 
	mbox = (cyg_mbox *) cyg_mempool_var_try_alloc(var_handle, sizeof(cyg_mbox));
	if (!mbox)
	    return SYS_MBOX_NULL;
	cyg_mbox_create(&handle, mbox);
 
#if SYS_STATS
    lwip_stats.sys.mbox.used++;
    if (lwip_stats.sys.mbox.used > lwip_stats.sys.mbox.max)
        lwip_stats.sys.mbox.max = lwip_stats.sys.mbox.used;
#endif
 
	return handle;
}
 
//
// Deallocates a mailbox. If there are messages still present in the mailbox
// when the mailbox is deallocated, it is an indication of a programming error
// in lwIP and the developer should be notified.
//
void
sys_mbox_free(sys_mbox_t mbox)
{
    if (!mbox)
        return;
 
    if (cyg_mbox_peek(mbox))
        LWIP_DEBUGF(SYS_DEBUG | LWIP_DBG_LEVEL_WARNING,
                    ("sys_mbox_free: mbox not empty\n"));
 
	cyg_mbox_delete(mbox);
	cyg_mempool_var_free(var_handle, (void *) mbox);
 
#if SYS_STATS
    lwip_stats.sys.mbox.used--;
#endif
}
 
//
// cyg_mbox_put() should not be passed a NULL, otherwise the cyg_mbox_get()
// will not know if it's real data or an error condition. But lwIP does pass
// NULL on occasion, in cases when maybe using a semaphore would be better. So
// this null_msg replaces NULL data.
//
static int null_msg;
 
//
// Posts the "msg" to the mailbox. This function have to block until the "msg"
// is really posted.
//
void
sys_mbox_post(sys_mbox_t mbox, void *msg)
{
    // Map NULL messages
    if (!msg)
        msg = &null_msg;
    while (cyg_mbox_put(mbox, msg) == false);
}
 
//
// Try to post the "msg" to the mailbox. Returns ERR_MEM if this one is full,
// else, ERR_OK if the "msg" is posted.
//
err_t
sys_mbox_trypost(sys_mbox_t mbox, void *msg)
{
    // Map NULL messages
    if (!msg)
        msg = &null_msg;
    return cyg_mbox_tryput(mbox, msg) ? ERR_OK : ERR_MEM;
}
 
//
// Blocks the thread until a message arrives in the mailbox, but does not block
// the thread longer than "timeout" milliseconds (similar to the
// sys_arch_sem_wait() function). If "timeout" is 0, the thread should be
// blocked until a message arrives. The "msg" argument is a result parameter
// that is set by the function (i.e., by doing "*msg = ptr"). The "msg"
// parameter maybe NULL to indicate that the message should be dropped.
//
// The return values are the same as for the sys_arch_sem_wait() function:
// Number of milliseconds spent waiting or SYS_ARCH_TIMEOUT if there was a
// timeout.
//
// Note that a function with a similar name, sys_mbox_fetch(), is implemented
// by lwIP.
//
u32_t
sys_arch_mbox_fetch(sys_mbox_t mbox, void **msg, u32_t timeout)
{
    void *m;
 
    if (timeout) {
        cyg_tick_count_t start_time = cyg_current_time();
 
        // Wait for mailbox with timeout
        if (!(m = cyg_mbox_timed_get(mbox, start_time + MS_TO_TICKS(timeout))))
            return SYS_ARCH_TIMEOUT;
        // Map NULL messages
        if (m == &null_msg)
            m = NULL;
        *msg = m;
        // Return elapsed time
        return TICKS_TO_MS(cyg_current_time() - start_time);
    } else {
        // Wait for semaphore indefinitely
        m = cyg_mbox_get(mbox);
        // Map NULL messages
        if (m == &null_msg)
            m = NULL;
        *msg = m;
        return 0;
    }
}
 
//
// This is similar to sys_arch_mbox_fetch, however if a message is not present
// in the mailbox, it immediately returns with the code SYS_MBOX_EMPTY. On
// success 0 is returned.
//
// To allow for efficient implementations, this can be defined as a
// function-like macro in sys_arch.h instead of a normal function. For example,
// a naive implementation could be:
//
// #define sys_arch_mbox_tryfetch(mbox,msg) sys_arch_mbox_fetch(mbox,msg,1)
//
// although this would introduce unnecessary delays.
//
u32_t
sys_arch_mbox_tryfetch(sys_mbox_t mbox, void **msg)
{
    void *m;
 
    m = cyg_mbox_tryget(mbox);
    if (!m)
        return SYS_MBOX_EMPTY;
 
    if (m == &null_msg)
        m = NULL;
    *msg = m;
 
    return 0;
}
 
//
// Returns a pointer to the per-thread sys_timeouts structure. In lwIP, each
// thread has a list of timeouts which is repressented as a linked list of
// sys_timeout structures. The sys_timeouts structure holds a pointer to a
// linked list of timeouts. This function is called by the lwIP timeout
// scheduler and must not return a NULL value.
//
// In a single thread sys_arch implementation, this function will simply return
// a pointer to a global sys_timeouts variable stored in the sys_arch module.
//
struct sys_timeouts *
sys_arch_timeouts(void)
{
    cyg_handle_t handle;
    struct lwip_thread *t;
 
    handle = cyg_thread_self();
    for (t = threads; t; t = t->next)
        if (t->handle == handle)
            return &(t->to);
 
    return &to;
}
 
//
// Starts a new thread named "name" with priority "prio" that will begin its
// execution in the function "thread()". The "arg" argument will be passed as
// an argument to the thread() function. The stack size to used for this thread
// is the "stacksize" parameter. The id of the new thread is returned. Both the
// id and the priority are system dependent.
//
sys_thread_t
sys_thread_new(char *name, void (* thread)(void *arg), void *arg,
               int stacksize, int prio)
{
    void *stack;
 
    cyg_mutex_lock(&stack_mutex);
    stack = stack_pos;
    stack_pos += stacksize;
    cyg_mutex_unlock(&stack_mutex);
 
    if (stack_pos > stack_data + TOTAL_STACKSIZE)
        CYG_FAIL("Not enough memory to allocate the thread's stack. You may "
                 "want to use cyg_lwip_thread_new() instead of "
                 "sys_thread_new() so you can provide external stack memory.");
 
    return cyg_lwip_thread_new(name, thread, arg, stack, stacksize, prio);
}
 
//
// Basically implements the sys_thread_new() call, but adds a "stack" parameter,
// allowing clients to provide their own stack buffers.
//
sys_thread_t
cyg_lwip_thread_new(char *name, void (* thread)(void *arg), void *arg,
                    void *stack, int stacksize, int prio)
{
    struct lwip_thread *t;
 
    t = (struct lwip_thread *) cyg_mempool_var_alloc(
            var_handle, sizeof(struct lwip_thread));
 
    t->next = threads;
    t->to.next = NULL;
 
    threads = t;
 
    cyg_thread_create(
        prio,
        (cyg_thread_entry_t *) thread,
        (cyg_addrword_t) arg,
        name,
        stack,
        stacksize,
        &t->handle,
        &t->thread
    );
    cyg_thread_resume(t->handle);
 
    return t->handle;
}
 
#endif // !NO_SYS
 
//
// Returns the current time in milliseconds.
//
u32_t
sys_now(void)
{
    return cyg_current_time() * MS_PER_TICK;
}
 
//
// This optional function does a "fast" critical region protection and returns
// the previous protection level. This function is only called during very short
// critical regions. An embedded system which supports ISR-based drivers might
// want to implement this function by disabling interrupts. Task-based systems
// might want to implement this by using a mutex or disabling tasking. This
// function should support recursive calls from the same task or interrupt. In
// other words, sys_arch_protect() could be called while already protected. In
// that case the return value indicates that it is already protected.
//
// sys_arch_protect() is only required if your port is supporting an operating
// system.
//
sys_prot_t
sys_arch_protect(void)
{
    cyg_scheduler_lock();
 
    return 0;
}
 
//
// This optional function does a "fast" set of critical region protection to the
// value specified by pval. See the documentation for sys_arch_protect() for
// more information. This function is only required if your port is supporting
// an operating system.
//
void
sys_arch_unprotect(sys_prot_t pval)
{
    LWIP_UNUSED_ARG(pval);
 
    cyg_scheduler_unlock();
}
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.