2011-05-14 12:19:48 -05:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
*
|
|
|
|
* @addtogroup PIOS PIOS Core hardware abstraction layer
|
|
|
|
* @{
|
|
|
|
* @addtogroup PIOS_FLASHFS_OBJLIST Object list based flash filesystem (low ram)
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file pios_flashfs_objlist.c
|
|
|
|
* @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2010.
|
|
|
|
* @brief A file system for storing UAVObject in flash chip
|
|
|
|
* @see The GNU Public License (GPL) Version 3
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
2011-07-04 21:44:41 -05:00
|
|
|
/*
|
|
|
|
* This program 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 3 of the License, or
|
2011-05-14 12:19:48 -05:00
|
|
|
* (at your option) any later version.
|
2011-07-04 21:44:41 -05:00
|
|
|
*
|
|
|
|
* This program 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
|
2011-05-14 12:19:48 -05:00
|
|
|
* for more details.
|
2011-07-04 21:44:41 -05:00
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
2011-05-14 12:19:48 -05:00
|
|
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include "openpilot.h"
|
|
|
|
#include "uavobjectmanager.h"
|
|
|
|
|
|
|
|
// Private functions
|
2011-07-15 15:30:10 -05:00
|
|
|
static int32_t PIOS_FLASHFS_ClearObjectTableHeader();
|
2011-05-14 12:19:48 -05:00
|
|
|
static int32_t PIOS_FLASHFS_GetObjAddress(uint32_t objId, uint16_t instId);
|
|
|
|
static int32_t PIOS_FLASHFS_GetNewAddress(uint32_t objId, uint16_t instId);
|
|
|
|
|
|
|
|
// Private variables
|
|
|
|
static int32_t numObjects = -1;
|
|
|
|
|
|
|
|
// Private structures
|
|
|
|
// Header for objects in the file system table
|
|
|
|
struct objectHeader {
|
|
|
|
uint32_t objMagic;
|
|
|
|
uint32_t objId;
|
|
|
|
uint32_t instId;
|
|
|
|
uint32_t address;
|
|
|
|
} __attribute__((packed));;
|
|
|
|
|
|
|
|
struct fileHeader {
|
|
|
|
uint32_t id;
|
|
|
|
uint16_t instId;
|
|
|
|
uint16_t size;
|
|
|
|
} __attribute__((packed));
|
|
|
|
|
|
|
|
|
2011-12-10 14:11:58 -06:00
|
|
|
#define OBJECT_TABLE_MAGIC 0x85FB3D35
|
|
|
|
#define OBJ_MAGIC 0x3015A371
|
2011-05-14 14:23:23 -05:00
|
|
|
#define OBJECT_TABLE_START 0x00000010
|
2011-11-26 14:55:16 -06:00
|
|
|
#define OBJECT_TABLE_END 0x00010000
|
|
|
|
#define SECTOR_SIZE 0x00010000
|
2011-10-07 14:08:56 -05:00
|
|
|
#define MAX_BADMAGIC 1000
|
2011-05-14 12:19:48 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize the flash object setting FS
|
|
|
|
* @return 0 if success, -1 if failure
|
|
|
|
*/
|
|
|
|
int32_t PIOS_FLASHFS_Init()
|
|
|
|
{
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 14:23:23 -05:00
|
|
|
// Check for valid object table or create one
|
|
|
|
uint32_t object_table_magic;
|
2011-10-07 14:08:56 -05:00
|
|
|
uint32_t magic_fail_count = 0;
|
2011-07-15 15:30:10 -05:00
|
|
|
bool magic_good = false;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
|
|
|
while(!magic_good) {
|
|
|
|
if (PIOS_Flash_W25X_ReadData(0, (uint8_t *)&object_table_magic, sizeof(object_table_magic)) != 0)
|
2011-05-14 14:23:23 -05:00
|
|
|
return -1;
|
2011-07-04 21:44:41 -05:00
|
|
|
if(object_table_magic != OBJECT_TABLE_MAGIC) {
|
|
|
|
if(magic_fail_count++ > MAX_BADMAGIC) {
|
2011-07-15 15:30:10 -05:00
|
|
|
PIOS_FLASHFS_ClearObjectTableHeader();
|
2011-07-15 18:06:18 -05:00
|
|
|
PIOS_LED_Toggle(LED1);
|
|
|
|
magic_fail_count = 0;
|
|
|
|
magic_good = true;
|
2011-07-04 21:44:41 -05:00
|
|
|
} else {
|
2011-10-07 14:08:56 -05:00
|
|
|
PIOS_DELAY_WaituS(1000);
|
2011-07-04 21:44:41 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
magic_good = true;
|
|
|
|
}
|
|
|
|
|
2011-05-14 14:23:23 -05:00
|
|
|
}
|
2011-05-14 16:11:30 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
int32_t addr = OBJECT_TABLE_START;
|
|
|
|
struct objectHeader header;
|
2011-05-14 14:23:23 -05:00
|
|
|
numObjects = 0;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
// Loop through header area while objects detect to count how many saved
|
2011-05-14 16:11:30 -05:00
|
|
|
while(addr < OBJECT_TABLE_END) {
|
2011-05-14 12:19:48 -05:00
|
|
|
// Read the instance data
|
|
|
|
if (PIOS_Flash_W25X_ReadData(addr, (uint8_t *)&header, sizeof(header)) != 0)
|
|
|
|
return -1;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
|
|
|
// Counting number of valid headers
|
|
|
|
if(header.objMagic != OBJ_MAGIC)
|
2011-05-14 16:11:30 -05:00
|
|
|
break;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 16:11:30 -05:00
|
|
|
numObjects++;
|
|
|
|
addr += sizeof(header);
|
2011-05-14 14:23:23 -05:00
|
|
|
}
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 14:23:23 -05:00
|
|
|
return 0;
|
2011-05-14 12:19:48 -05:00
|
|
|
}
|
|
|
|
|
2011-08-15 10:24:37 -05:00
|
|
|
/**
|
|
|
|
* @brief Erase the whole flash chip and create the file system
|
|
|
|
* @return 0 if successful, -1 if not
|
|
|
|
*/
|
|
|
|
int32_t PIOS_FLASHFS_Format()
|
|
|
|
{
|
|
|
|
if(PIOS_Flash_W25X_EraseChip() != 0)
|
|
|
|
return -1;
|
|
|
|
if(PIOS_FLASHFS_ClearObjectTableHeader() != 0)
|
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
/**
|
|
|
|
* @brief Erase the headers for all objects in the flash chip
|
|
|
|
* @return 0 if successful, -1 if not
|
|
|
|
*/
|
2011-07-15 15:30:10 -05:00
|
|
|
static int32_t PIOS_FLASHFS_ClearObjectTableHeader()
|
2011-05-14 12:19:48 -05:00
|
|
|
{
|
2011-06-11 22:31:51 -05:00
|
|
|
if(PIOS_Flash_W25X_EraseSector(0) != 0)
|
2011-05-14 12:19:48 -05:00
|
|
|
return -1;
|
2011-05-14 14:23:23 -05:00
|
|
|
|
|
|
|
uint32_t object_table_magic = OBJECT_TABLE_MAGIC;
|
|
|
|
if (PIOS_Flash_W25X_WriteData(0, (uint8_t *)&object_table_magic, sizeof(object_table_magic)) != 0)
|
|
|
|
return -1;
|
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Get the address of an object
|
|
|
|
* @param obj UAVObjHandle for that object
|
|
|
|
* @parma instId Instance id for that object
|
2011-05-14 14:23:23 -05:00
|
|
|
* @return address if successful, -1 if not found
|
2011-05-14 12:19:48 -05:00
|
|
|
*/
|
2011-07-04 21:44:41 -05:00
|
|
|
static int32_t PIOS_FLASHFS_GetObjAddress(uint32_t objId, uint16_t instId)
|
2011-05-14 12:19:48 -05:00
|
|
|
{
|
|
|
|
int32_t addr = OBJECT_TABLE_START;
|
|
|
|
struct objectHeader header;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
// Loop through header area while objects detect to count how many saved
|
2011-05-14 16:11:30 -05:00
|
|
|
while(addr < OBJECT_TABLE_END) {
|
2011-05-14 12:19:48 -05:00
|
|
|
// Read the instance data
|
|
|
|
if (PIOS_Flash_W25X_ReadData(addr, (uint8_t *) &header, sizeof(header)) != 0)
|
|
|
|
return -1;
|
2011-07-04 21:44:41 -05:00
|
|
|
if(header.objMagic != OBJ_MAGIC)
|
2011-05-14 16:11:30 -05:00
|
|
|
break; // stop searching once hit first non-object header
|
2011-05-14 12:19:48 -05:00
|
|
|
else if (header.objId == objId && header.instId == instId)
|
|
|
|
break;
|
2011-05-14 16:11:30 -05:00
|
|
|
addr += sizeof(header);
|
2011-05-14 12:19:48 -05:00
|
|
|
}
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 16:11:30 -05:00
|
|
|
if (header.objId == objId && header.instId == instId)
|
2011-05-14 12:19:48 -05:00
|
|
|
return header.address;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 14:23:23 -05:00
|
|
|
return -1;
|
2011-05-14 12:19:48 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Returns an address for a new object and creates entry into object table
|
|
|
|
* @param[in] obj Object handle for object to be saved
|
|
|
|
* @param[in] instId The instance id of object to be saved
|
|
|
|
* @return 0 if success or error code
|
|
|
|
* @retval -1 Object not found
|
2011-07-04 21:44:41 -05:00
|
|
|
* @retval -2 No room in object table
|
2011-05-14 12:19:48 -05:00
|
|
|
* @retval -3 Unable to write entry into object table
|
|
|
|
* @retval -4 FS not initialized
|
|
|
|
*/
|
|
|
|
int32_t PIOS_FLASHFS_GetNewAddress(uint32_t objId, uint16_t instId)
|
|
|
|
{
|
|
|
|
struct objectHeader header;
|
|
|
|
|
|
|
|
if(numObjects < 0)
|
|
|
|
return -4;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
// Don't worry about max size of flash chip here, other code will catch that
|
|
|
|
header.objMagic = OBJ_MAGIC;
|
|
|
|
header.objId = objId;
|
|
|
|
header.instId = instId;
|
2011-05-14 16:11:30 -05:00
|
|
|
header.address = OBJECT_TABLE_END + SECTOR_SIZE * numObjects;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 16:11:30 -05:00
|
|
|
int32_t addr = OBJECT_TABLE_START + sizeof(header) * numObjects;
|
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
// No room for this header in object table
|
2011-07-04 21:44:41 -05:00
|
|
|
if((addr + sizeof(header)) > OBJECT_TABLE_END)
|
|
|
|
return -2;
|
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
if(PIOS_Flash_W25X_WriteData(addr, (uint8_t *) &header, sizeof(header)) != 0)
|
|
|
|
return -3;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
// This numObejcts value must stay consistent or there will be a break in the table
|
|
|
|
// and later the table will have bad values in it
|
2011-07-04 21:44:41 -05:00
|
|
|
numObjects++;
|
|
|
|
return header.address;
|
2011-05-14 12:19:48 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Saves one object instance per sector
|
|
|
|
* @param[in] obj UAVObjHandle the object to save
|
|
|
|
* @param[in] instId The instance of the object to save
|
|
|
|
* @return 0 if success or -1 if failure
|
|
|
|
* @note This uses one sector on the flash chip per object so that no buffering in ram
|
|
|
|
* must be done when erasing the sector before a save
|
|
|
|
*/
|
2011-07-04 21:44:41 -05:00
|
|
|
int32_t PIOS_FLASHFS_ObjSave(UAVObjHandle obj, uint16_t instId, uint8_t * data)
|
2011-05-14 12:19:48 -05:00
|
|
|
{
|
|
|
|
uint32_t objId = UAVObjGetID(obj);
|
2011-06-04 09:41:47 -05:00
|
|
|
uint8_t crc = 0;
|
2011-05-14 12:19:48 -05:00
|
|
|
|
|
|
|
int32_t addr = PIOS_FLASHFS_GetObjAddress(objId, instId);
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
// Object currently not saved
|
2011-07-04 21:44:41 -05:00
|
|
|
if(addr < 0)
|
2011-05-14 12:19:48 -05:00
|
|
|
addr = PIOS_FLASHFS_GetNewAddress(objId, instId);
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
// Could not allocate a sector
|
|
|
|
if(addr < 0)
|
|
|
|
return -1;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
struct fileHeader header = {
|
|
|
|
.id = objId,
|
|
|
|
.instId = instId,
|
2011-05-14 14:23:23 -05:00
|
|
|
.size = UAVObjGetNumBytes(obj)
|
2011-05-14 12:19:48 -05:00
|
|
|
};
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 14:28:11 -05:00
|
|
|
if(PIOS_Flash_W25X_EraseSector(addr) != 0)
|
|
|
|
return -2;
|
2011-05-14 12:19:48 -05:00
|
|
|
|
|
|
|
// Save header
|
2011-07-04 21:44:41 -05:00
|
|
|
// This information IS redundant with the object table id. Oh well. Better safe than sorry.
|
2011-05-14 14:28:11 -05:00
|
|
|
if(PIOS_Flash_W25X_WriteData(addr, (uint8_t *) &header, sizeof(header)) != 0)
|
|
|
|
return -3;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-06-04 09:41:47 -05:00
|
|
|
// Update CRC
|
|
|
|
crc = PIOS_CRC_updateCRC(0, (uint8_t *) &header, sizeof(header));
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
// Save data
|
2011-05-14 14:28:11 -05:00
|
|
|
if(PIOS_Flash_W25X_WriteData(addr + sizeof(header), data, UAVObjGetNumBytes(obj)) != 0)
|
|
|
|
return -4;
|
2011-06-04 09:41:47 -05:00
|
|
|
|
|
|
|
// Update CRC
|
|
|
|
crc = PIOS_CRC_updateCRC(crc, (uint8_t *) data, UAVObjGetNumBytes(obj));
|
|
|
|
|
|
|
|
// Save CRC (written so will work when CRC changes to uint16)
|
|
|
|
if(PIOS_Flash_W25X_WriteData(addr + sizeof(header) + UAVObjGetNumBytes(obj), (uint8_t *) &crc, sizeof(crc)) != 0)
|
|
|
|
return -4;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 14:23:23 -05:00
|
|
|
return 0;
|
2011-05-14 12:19:48 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Load one object instance per sector
|
|
|
|
* @param[in] obj UAVObjHandle the object to save
|
|
|
|
* @param[in] instId The instance of the object to save
|
|
|
|
* @return 0 if success or error code
|
|
|
|
* @retval -1 if object not in file table
|
2011-05-14 14:28:11 -05:00
|
|
|
* @retval -2 if unable to retrieve object header
|
|
|
|
* @retval -3 if loaded data instId or objId don't match
|
|
|
|
* @retval -4 if unable to retrieve instance data
|
2011-06-04 09:41:47 -05:00
|
|
|
* @retval -5 if unable to read CRC
|
|
|
|
* @retval -6 if CRC doesn't match
|
2011-05-14 12:19:48 -05:00
|
|
|
* @note This uses one sector on the flash chip per object so that no buffering in ram
|
|
|
|
* must be done when erasing the sector before a save
|
|
|
|
*/
|
|
|
|
int32_t PIOS_FLASHFS_ObjLoad(UAVObjHandle obj, uint16_t instId, uint8_t * data)
|
|
|
|
{
|
|
|
|
uint32_t objId = UAVObjGetID(obj);
|
2011-06-05 09:05:33 -05:00
|
|
|
uint16_t objSize = UAVObjGetNumBytes(obj);
|
2011-06-04 09:41:47 -05:00
|
|
|
uint8_t crc = 0;
|
|
|
|
uint8_t crcFlash = 0;
|
2011-06-05 09:05:33 -05:00
|
|
|
const uint8_t crc_read_step = 8;
|
|
|
|
uint8_t crc_read_buffer[crc_read_step];
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
int32_t addr = PIOS_FLASHFS_GetObjAddress(objId, instId);
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
// Object currently not saved
|
2011-07-04 21:44:41 -05:00
|
|
|
if(addr < 0)
|
2011-05-14 12:19:48 -05:00
|
|
|
return -1;
|
2011-05-14 16:11:30 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
struct fileHeader header;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
// Load header
|
|
|
|
// This information IS redundant with the object table id. Oh well. Better safe than sorry.
|
2011-05-14 14:28:11 -05:00
|
|
|
if(PIOS_Flash_W25X_ReadData(addr, (uint8_t *) &header, sizeof(header)) != 0)
|
|
|
|
return -2;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-06-04 09:41:47 -05:00
|
|
|
// Update CRC
|
|
|
|
crc = PIOS_CRC_updateCRC(0, (uint8_t *) &header, sizeof(header));
|
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
if((header.id != objId) || (header.instId != instId))
|
2011-05-14 14:28:11 -05:00
|
|
|
return -3;
|
2011-06-04 09:41:47 -05:00
|
|
|
|
2011-06-05 09:05:33 -05:00
|
|
|
// To avoid having to allocate the RAM for a copy of the object, we read by chunks
|
|
|
|
// and compute the CRC
|
|
|
|
for(uint32_t i = 0; i < objSize; i += crc_read_step) {
|
|
|
|
PIOS_Flash_W25X_ReadData(addr + sizeof(header) + i, crc_read_buffer, crc_read_step);
|
|
|
|
uint8_t valid_bytes = ((i + crc_read_step) >= objSize) ? objSize - i : crc_read_step;
|
|
|
|
crc = PIOS_CRC_updateCRC(crc, crc_read_buffer, valid_bytes);
|
2011-06-04 09:41:47 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Read CRC (written so will work when CRC changes to uint16)
|
2011-06-05 09:05:33 -05:00
|
|
|
if(PIOS_Flash_W25X_ReadData(addr + sizeof(header) + objSize, (uint8_t *) &crcFlash, sizeof(crcFlash)) != 0)
|
2011-06-04 09:41:47 -05:00
|
|
|
return -5;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-06-04 09:41:47 -05:00
|
|
|
if(crc != crcFlash)
|
|
|
|
return -6;
|
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
// Read the instance data
|
2011-06-05 09:05:33 -05:00
|
|
|
if (PIOS_Flash_W25X_ReadData(addr + sizeof(header), data, objSize) != 0)
|
2011-05-14 14:28:11 -05:00
|
|
|
return -4;
|
2011-06-04 09:41:47 -05:00
|
|
|
|
2011-05-14 14:23:23 -05:00
|
|
|
return 0;
|
2011-05-14 12:19:48 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Delete object from flash
|
|
|
|
* @param[in] obj UAVObjHandle the object to save
|
|
|
|
* @param[in] instId The instance of the object to save
|
|
|
|
* @return 0 if success or error code
|
|
|
|
* @retval -1 if object not in file table
|
|
|
|
* @retval -2 Erase failed
|
2011-07-04 21:44:41 -05:00
|
|
|
* @note To avoid buffering the file table (1k ram!) the entry in the file table
|
|
|
|
* remains but destination sector is erased. This will make the load fail as the
|
2011-05-14 12:19:48 -05:00
|
|
|
* file header won't match the object. At next save it goes back there.
|
|
|
|
*/
|
|
|
|
int32_t PIOS_FLASHFS_ObjDelete(UAVObjHandle obj, uint16_t instId)
|
|
|
|
{
|
|
|
|
uint32_t objId = UAVObjGetID(obj);
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
int32_t addr = PIOS_FLASHFS_GetObjAddress(objId, instId);
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 12:19:48 -05:00
|
|
|
// Object currently not saved
|
2011-07-04 21:44:41 -05:00
|
|
|
if(addr < 0)
|
2011-05-14 12:19:48 -05:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if(PIOS_Flash_W25X_EraseSector(addr) != 0)
|
|
|
|
return -2;
|
2011-07-04 21:44:41 -05:00
|
|
|
|
2011-05-14 14:23:23 -05:00
|
|
|
return 0;
|
2011-07-04 21:44:41 -05:00
|
|
|
}
|