/**************************************************************************** * libs/libc/misc/lib_pathbuffer.c * * 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. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include /**************************************************************************** * Pre-processor definitions ****************************************************************************/ /**************************************************************************** * Private Types ****************************************************************************/ struct pathbuffer_s { mutex_t lock; /* Lock for the buffer */ unsigned int free_bitmap; /* Bitmap of free buffer */ char buffer[CONFIG_LIBC_MAX_PATHBUFFER][PATH_MAX]; }; /**************************************************************************** * Private Data ****************************************************************************/ static struct pathbuffer_s g_pathbuffer = { NXMUTEX_INITIALIZER, (1u << CONFIG_LIBC_MAX_PATHBUFFER) - 1, }; /**************************************************************************** * Private Functions ****************************************************************************/ /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: lib_get_pathbuffer * * Description: * The lib_get_pathbuffer() function returns a pointer to a temporary * buffer. The buffer is allocated from a pool of pre-allocated buffers * and if the pool is exhausted, a new buffer is allocated through * kmm_malloc(). The size of the buffer is PATH_MAX, and must freed by * calling lib_put_pathbuffer(). * * Returned Value: * On success, lib_get_pathbuffer() returns a pointer to a temporary * buffer. On failure, NULL is returned. * ****************************************************************************/ FAR char *lib_get_pathbuffer(void) { int index; /* Try to find a free buffer */ nxmutex_lock(&g_pathbuffer.lock); index = ffs(g_pathbuffer.free_bitmap) - 1; if (index >= 0 && index < CONFIG_LIBC_MAX_PATHBUFFER) { g_pathbuffer.free_bitmap &= ~(1u << index); nxmutex_unlock(&g_pathbuffer.lock); return g_pathbuffer.buffer[index]; } nxmutex_unlock(&g_pathbuffer.lock); /* If no free buffer is found, allocate a new one if * CONFIG_LIBC_PATHBUFFER_MALLOC is enabled */ #ifdef CONFIG_LIBC_PATHBUFFER_MALLOC return lib_malloc(PATH_MAX); #else return NULL; #endif } /**************************************************************************** * Name: lib_put_pathbuffer * * Description: * The lib_put_pathbuffer() function frees a temporary buffer that was * allocated by lib_get_pathbuffer(). If the buffer was allocated * dynamically, it is freed by calling kmm_free(). Otherwise, the buffer * is marked as free in the pool of pre-allocated buffers. * * Returned Value: * None * ****************************************************************************/ void lib_put_pathbuffer(FAR char *buffer) { int index; nxmutex_lock(&g_pathbuffer.lock); index = (buffer - &g_pathbuffer.buffer[0][0]) / PATH_MAX; if (index >= 0 && index < CONFIG_LIBC_MAX_PATHBUFFER) { /* Mark the corresponding bit as free */ g_pathbuffer.free_bitmap |= 1u << index; nxmutex_unlock(&g_pathbuffer.lock); return; } nxmutex_unlock(&g_pathbuffer.lock); /* Free the buffer if it was dynamically allocated */ #ifdef CONFIG_LIBC_PATHBUFFER_MALLOC return lib_free(buffer); #endif }