#define ATR_MAX_ADAPTERS 100 /* Just a sanity limit */ #define ATR_MAX_SYMLINK_LEN 11 /* Longest name is 10 chars: "channel-99" */
/** * struct i2c_atr_alias_pair - Holds the alias assigned to a client address. * @node: List node * @addr: Address of the client on the child bus. * @alias: I2C alias address assigned by the driver. * This is the address that will be used to issue I2C transactions * on the parent (physical) bus. * @fixed: Alias pair cannot be replaced during dynamic address attachment. * This flag is necessary for situations where a single I2C transaction * contains more distinct target addresses than the ATR channel can handle. * It marks addresses that have already been attached to an alias so * that their alias pair is not evicted by a subsequent address in the same * transaction. *
*/ struct i2c_atr_alias_pair { struct list_head node; bool fixed;
u16 addr;
u16 alias;
};
/** * struct i2c_atr_alias_pool - Pool of client aliases available for an ATR. * @size: Total number of aliases * @shared: Indicates if this alias pool is shared by multiple channels * * @lock: Lock protecting @aliases and @use_mask * @aliases: Array of aliases, must hold exactly @size elements * @use_mask: Mask of used aliases
*/ struct i2c_atr_alias_pool {
size_t size; bool shared;
/** * struct i2c_atr_chan - Data for a channel. * @adap: The &struct i2c_adapter for the channel * @atr: The parent I2C ATR * @chan_id: The ID of this channel * @alias_pairs_lock: Mutex protecting @alias_pairs * @alias_pairs_lock_key: Lock key for @alias_pairs_lock * @alias_pairs: List of @struct i2c_atr_alias_pair containing the * assigned aliases * @alias_pool: Pool of available client aliases * * @orig_addrs_lock: Mutex protecting @orig_addrs * @orig_addrs_lock_key: Lock key for @orig_addrs_lock * @orig_addrs: Buffer used to store the original addresses during transmit * @orig_addrs_size: Size of @orig_addrs
*/ struct i2c_atr_chan { struct i2c_adapter adap; struct i2c_atr *atr;
u32 chan_id;
/** * struct i2c_atr - The I2C ATR instance * @parent: The parent &struct i2c_adapter * @dev: The device that owns the I2C ATR instance * @ops: &struct i2c_atr_ops * @priv: Private driver data, set with i2c_atr_set_driver_data() * @algo: The &struct i2c_algorithm for adapters * @lock: Lock for the I2C bus segment (see &struct i2c_lock_operations) * @lock_key: Lock key for @lock * @max_adapters: Maximum number of adapters this I2C ATR can have * @flags: Flags for ATR * @alias_pool: Optional common pool of available client aliases * @i2c_nb: Notifier for remote client add & del events * @adapter: Array of adapters
*/ struct i2c_atr { struct i2c_adapter *parent; struct device *dev; conststruct i2c_atr_ops *ops;
void *priv;
struct i2c_algorithm algo; /* lock for the I2C bus segment (see struct i2c_lock_operations) */ struct mutex lock; struct lock_class_key lock_key; int max_adapters;
u32 flags;
/* Must be called with alias_pairs_lock held */ staticstruct i2c_atr_alias_pair *i2c_atr_create_c2a(struct i2c_atr_chan *chan,
u16 alias, u16 addr)
{ struct i2c_atr_alias_pair *c2a;
lockdep_assert_held(&chan->alias_pairs_lock);
c2a = kzalloc(sizeof(*c2a), GFP_KERNEL); if (!c2a) return NULL;
c2a->addr = addr;
c2a->alias = alias;
list_add(&c2a->node, &chan->alias_pairs);
return c2a;
}
/* Must be called with alias_pairs_lock held */ staticvoid i2c_atr_destroy_c2a(struct i2c_atr_alias_pair **pc2a)
{
list_del(&(*pc2a)->node);
kfree(*pc2a);
*pc2a = NULL;
}
/* * Replace all message addresses with their aliases, saving the original * addresses. * * This function is internal for use in i2c_atr_master_xfer(). It must be * followed by i2c_atr_unmap_msgs() to restore the original addresses.
*/ staticint i2c_atr_map_msgs(struct i2c_atr_chan *chan, struct i2c_msg *msgs, int num)
{ struct i2c_atr *atr = chan->atr; staticstruct i2c_atr_alias_pair *c2a; int i, ret = 0;
/* Ensure we have enough room to save the original addresses */ if (unlikely(chan->orig_addrs_size < num)) {
u16 *new_buf;
/* We don't care about old data, hence no realloc() */
new_buf = kmalloc_array(num, sizeof(*new_buf), GFP_KERNEL); if (!new_buf) return -ENOMEM;
/* * Restore all message address aliases with the original addresses. This * function is internal for use in i2c_atr_master_xfer() and for this reason it * needs no null and size checks on orig_addr. * * @see i2c_atr_map_msgs()
*/ staticvoid i2c_atr_unmap_msgs(struct i2c_atr_chan *chan, struct i2c_msg *msgs, int num)
{ struct i2c_atr_alias_pair *c2a; int i;
for (i = 0; i < num; i++)
msgs[i].addr = chan->orig_addrs[i];
mutex_lock(&chan->alias_pairs_lock);
if (unlikely(list_empty(&chan->alias_pairs))) goto out_unlock;
// unfix c2a entries so that subsequent transfers can reuse their aliases
list_for_each_entry(c2a, &chan->alias_pairs, node) {
c2a->fixed = false;
}
client = i2c_verify_client(dev); if (!client) return NOTIFY_DONE;
/* Is the client in one of our adapters? */ for (chan_id = 0; chan_id < atr->max_adapters; ++chan_id) { if (client->adapter == atr->adapter[chan_id]) break;
}
if (chan_id == atr->max_adapters) return NOTIFY_DONE;
switch (event) { case BUS_NOTIFY_ADD_DEVICE:
ret = i2c_atr_attach_addr(client->adapter, client->addr); if (ret)
dev_err(atr->dev, "Failed to attach remote client '%s': %d\n",
dev_name(dev), ret); break;
case BUS_NOTIFY_REMOVED_DEVICE:
i2c_atr_detach_addr(client->adapter, client->addr); break;
if (!fwnode_property_present(dev_fwnode(dev), "i2c-alias-pool")) {
num_aliases = 0;
} else {
ret = fwnode_property_count_u32(dev_fwnode(dev), "i2c-alias-pool"); if (ret < 0) {
dev_err(dev, "Failed to count 'i2c-alias-pool' property: %d\n",
ret); return ret;
}
num_aliases = ret;
}
alias_pool = i2c_atr_alloc_alias_pool(num_aliases, true); if (IS_ERR(alias_pool)) {
ret = PTR_ERR(alias_pool);
dev_err(dev, "Failed to allocate alias pool, err %d\n", ret); return ret;
}
atr->alias_pool = alias_pool;
if (!alias_pool->size) return 0;
aliases32 = kcalloc(num_aliases, sizeof(*aliases32), GFP_KERNEL); if (!aliases32) {
ret = -ENOMEM; goto err_free_alias_pool;
}
ret = fwnode_property_read_u32_array(dev_fwnode(dev), "i2c-alias-pool",
aliases32, num_aliases); if (ret < 0) {
dev_err(dev, "Failed to read 'i2c-alias-pool' property: %d\n",
ret); goto err_free_aliases32;
}
for (i = 0; i < num_aliases; i++) { if (!(aliases32[i] & 0xffff0000)) {
alias_pool->aliases[i] = aliases32[i]; continue;
}
dev_err(dev, "Failed to parse 'i2c-alias-pool' property: I2C flags are not supported\n");
ret = -EINVAL; goto err_free_aliases32;
}
kfree(aliases32);
dev_dbg(dev, "i2c-alias-pool has %zu aliases\n", alias_pool->size);
ret = sysfs_create_link(&chan->adap.dev.kobj, &dev->kobj, "atr_device"); if (ret)
dev_warn(dev, "can't create symlink to atr device\n");
ret = sysfs_create_link(&dev->kobj, &chan->adap.dev.kobj, symlink_name); if (ret)
dev_warn(dev, "can't create symlink for channel %u\n", chan_id);
dev_dbg(dev, "Added ATR child bus %d\n", i2c_adapter_id(&chan->adap));
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.