/* * Author: Jon Trulson <jtrulson@ics.com> * Copyright (c) 2015 Intel Corporation. * * * This code was adapted from the Seeed Studio code at: * https://github.com/Seeed-Studio/NFC_Tag_M24LR6E * * Copyright (c) 2014 seeed technology inc. * Website : www.seeed.cc * Author : lawliet zou * Create Time: March 2014 * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #pragma once #include <string> #include <mraa/common.hpp> #include <mraa/i2c.hpp> #define M24LR64E_I2C_BUS 0 #define M24LR64E_DEFAULT_I2C_ADDR 0x53 #define M24LR64E_DEFAULT_I2C_ADDR_E2 (M24LR64E_DEFAULT_I2C_ADDR | 0x04) namespace upm { /** * @brief Grove NFC Tag * @defgroup m24lr64e libupm-m24lr64e * @ingroup seeed i2c other */ /** * @library m24lr64e * @sensor m24lr64e * @comname Grove NFC Tag * @type other * @man seeed * @web http://www.seeedstudio.com/wiki/Grove_-_NFC_Tag * @con i2c * * @brief C++ API for the M24LR64E-based Grove NFC Tag * * Grove NFC tag is an 8KB electrically erasable programmable read-only memory (EEPROM) * that can be written to or read from using I2C and NFC-equipped devices. * * The user mode (default) allows read and write access to all 8KB * of space, provided the sector security status (SSS) allows it. * The root mode allows modification of the SSS data and other * information, provided the proper password is submitted. The * default password for a new tag is 0x00000000. See the datasheet * for more details. * * The Seeed Studio* wiki page for this device includes a link to an * Android* application that can be used to also read and write the * device via NFC, as well as set NFC passwords, which cannot be * done via I2C. * * @image html m24lr64e.jpg * @snippet m24lr64e.cxx Interesting */ class M24LR64E { public: static const int EEPROM_I2C_LENGTH = 8192; static const int PASSWORD_LENGTH = 4; static const int SECTOR_SECURITY_STATUS_BASE_ADDR = 0x800; // 2048 static const uint8_t LOCK_PROTECT_BIT = 0x01; static const uint8_t WRITE_READ_PROTECT_BIT = 0x02; static const uint8_t PASSWORD_CTRL_BIT = 0x04; static const int UID_LENGTH = 8; // bytes static const unsigned int I2C_WRITE_TIME = 5; // 5ms /** * M24LR64E addresses, accessible only in the root mode */ typedef enum { I2C_PASSWORD_ADDR = 2304, RF_PASSWORD_1_ADDR = 2308, // RF pwds not available in RF_PASSWORD_2_ADDR = 2312, // I2C access modes RF_PASSWORD_3_ADDR = 2316, DSFID_ADDR = 2320, // 1 byte AFI_ADDR = 2321, // 1 byte RESV_ADDR = 2322, // 1 bytes CONFIG_ADDR = 2323, // 1 bytes UID_ADDR = 2324, // 8 bytes MEM_SIZE_ADDR = 2332, // 3 bytes IC_REF_ADDR = 2335, // 1 byte PROG_COMP_ENERGY_HARVEST_ADDR = 2339 // 1 byte } M24LR64E_ADDR_T; enum AccessMode { USER_MODE = 0x0, // offers simple read/write access right ROOT_MODE = 0x1 // offers password change access right }; enum SectorAccessRight { // ********************************** // * submit passWd * no submit * //b2,b1 * Read * Write * Read * Write * // 00 * 1 1 1 0 * // 01 * 1 1 1 1 * // 10 * 1 1 0 0 * // 11 * 0 1 0 0 * // ********************************** Access_1110 = 0, Access_1111 = 1, Access_1100 = 2, Access_0111 = 3, }; enum SectorSelectPassWd { //00 => no passwd protect //01 => passWd 1 //10 => passWd 2 //11 => passwd 3 noPasswd = 0, passwd_1 = 1, passwd_2 = 2, passwd_3 = 3, }; /** * M24LR64E constructor * * @param bus I2C bus to use * @param mode Access mode (user or root) to use */ M24LR64E(int bus, AccessMode mode = USER_MODE); /** * M24LR64E destructor */ ~M24LR64E(); /** * Submits an I2C access password * * @param passwd 4-byte access password */ bool submitPasswd(uint32_t passwd); /** * Writes a new I2C password * * @param passwd 4-byte access password */ bool writePasswd(uint32_t passwd); /** * Sets a protection bit for a sector. Must be in the root mode * * @param sectorNumber Sector whose protection you are modifying * @param protectEnable True if you are enabling protection * @param accessRight Access rights to set * @param passwd Password number to enable, if any */ void sectorProtectConfig(unsigned int sectorNumber, bool protectEnable, SectorAccessRight accessRight, SectorSelectPassWd passwd); /** * Clears sector protection bits. Must be in the root mode. */ void clearSectorProtect(void); /** * Sets or clears a sector security status lock bit for a sector. * Must be in the root mode. * * @param sectorNumber Sector whose SSS you want to modify * @param sockEnable True to set the bit, false to clear it */ void sectorWriteLockBit(unsigned int sectorNumber, bool sockEnable); /** * Returns a data storage family identifier (DSFID) * Must be in the root mode. * * @return DSFID */ uint8_t getDSFID(); /** * Returns an application family identifier (AFI) * Must be in the root mode. * * @return AFI */ uint8_t getAFI(); /** * Returns a unique ID. * Must be in the root mode. * Maintained to preserve compatibility with older code. * * @result buf Buffer to hold the UID. Must be UID_LENGTH bytes. */ uint8_t *getUID(); /** * Returns the memory size * Must be in the root mode. * * @return Amount of memory present */ uint32_t getMemorySize(); /** * Sets all memory to 0, if permissions allow */ void clearMemory(); /** * Writes a byte to the EEPROM * * @param address Address to write to * @param data Data to write */ mraa::Result writeByte(unsigned int address, uint8_t data); /** * Writes bytes to the EEPROM * * @param address Address to write to * @param data Data to write * @param data Length of the data buffer */ mraa::Result writeBytes(unsigned int address, uint8_t* buffer, int len); /** * Reads a byte from the EEPROM * * @param address Address to read from * @return data Value read */ uint8_t readByte(unsigned int address); /** * Reads multiple bytes from the EEPROM * * @param address Address to read from * @param buffer Buffer to store data * @param len Number of bytes to read */ int readBytes(unsigned int address, uint8_t* buffer, int len); protected: mraa::I2c m_i2c; mraa::Result EEPROM_Write_Byte(unsigned int address, uint8_t data); mraa::Result EEPROM_Write_Bytes(unsigned int address, uint8_t* data, int len); uint8_t EEPROM_Read_Byte(unsigned int address); int EEPROM_Read_Bytes(unsigned int address, uint8_t* buffer, int len); private: uint8_t m_addr; }; }