#ifndef _PCILIB_REGISTER_H
#define _PCILIB_REGISTER_H

#include <uthash.h>

#include <pcilib.h>
#include <pcilib/bank.h>

#define PCILIB_REGISTER_NO_BITS			0
#define PCILIB_REGISTER_ALL_BITS		((pcilib_register_value_t)-1)


typedef enum {
    PCILIB_REGISTER_STANDARD = 0,               /**< Standard register */
    PCILIB_REGISTER_FIFO,                       /**< FIFO register */
    PCILIB_REGISTER_BITS,                       /**< Besides a big standard register, the register bit-fields may be described by bit registers */
    PCILIB_REGISTER_PROPERTY                    /**< A special register bound to a property and gettings/setting it value through it */
} pcilib_register_type_t;

typedef struct {
    const char *name;
    const char *view;
} pcilib_view_reference_t;

typedef struct {
    pcilib_register_addr_t addr;		/**< Register address in the bank */
    pcilib_register_size_t offset;		/**< Register offset in the byte (in bits) */
    pcilib_register_size_t bits;		/**< Register size in bits */
    pcilib_register_value_t defvalue;		/**< Default register value (some protocols, i.e. software registers, may set it during the initialization) */
    pcilib_register_value_t rwmask;		/**< Used to define how external bits of PCILIB_REGISTER_BITS registers are treated. 
						To keep bit value unchanged, we need to observe the following behavior depending on status of corresponding bit in this field:
						1 - standard bit (i.e. if we want to keep the bit value we need to read it, and, the write back), 
						0 - non-standard bit which behavior is defined by mode (only partially implemented. 
						    so far only 1C/1I modes (zero should be written to preserve the value) are supported */
    pcilib_register_mode_t mode;		/**< Register access (ro/wo/rw) and how writting to register works (if value just set as specified or, for instance, the bits which
						are on in the value are cleared/inverted). For information only, no preprocessing on bits is performed. */
    pcilib_register_type_t type;		/**< Defines type of register is it standard register, subregister for bit fields or view, fifo */
    pcilib_register_bank_addr_t bank;		/**< Specified the address of the bank this register belongs to */

    const char *name;				/**< The access name of the register */
    const char *description;			/**< Brief description of the register */

    pcilib_view_reference_t *views;		/**< List of supported views for this register */
} pcilib_register_description_t;

typedef struct {
    const char *name;                                                                   /**< Register name */
    pcilib_register_t reg;                                                              /**< Register index */
    pcilib_register_bank_t bank;							/**< Reference to bank containing the register */
    pcilib_register_value_range_t range;						/**< Minimum & maximum allowed values */
    pcilib_xml_node_t *xml;								/**< Additional XML properties */
    pcilib_view_reference_t *views;							/**< For non-static list of views, this vairables holds a copy of a NULL-terminated list from model (if present, memory should be de-allocated) */
    UT_hash_handle hh;
} pcilib_register_context_t;

#ifdef __cplusplus
extern "C" {
#endif

/**
 * Use this function to add new registers into the model. Currently, it is considered a error
 * to re-add already defined register. If it will turn out to be useful to redefine some registers 
 * from the model, it may change in the future. However, we should think how to treat bit-registers
 * in this case. The function will copy the context of registers structure, but name, 
 * description, and other strings in the structure are considered to have static duration 
 * and will not be copied. On error no new registers are initalized.
 * @param[in,out] ctx - pcilib context
 * @param[in] flags - not used now, but in future may instruct if existing registers should be reported as error (default), overriden or ignored
 * @param[in] n - number of registers to initialize. It is OK to pass 0 if registers array is NULL terminated (last member of the array have all members set to 0)
 * @param[in] registers - register descriptions
 * @param[out] ids - if specified will contain the ids of the newly registered and overriden registers
 * @return - error or 0 on success
 */
int pcilib_add_registers(pcilib_t *ctx, pcilib_model_modification_flags_t flags, size_t n, const pcilib_register_description_t *registers, pcilib_register_t *ids);

/**
 * Destroys data associated with registers. This is an internal function and will
 * be called during clean-up.
 * @param[in,out] ctx - pcilib context
 * @param[in] start - specifies first register to clean (used to clean only part of the registers to keep the defined state if pcilib_add_registers has failed)
 */
void pcilib_clean_registers(pcilib_t *ctx, pcilib_register_t start);

#ifdef __cplusplus
}
#endif

#endif /* _PCILIB_REGISTER_H */