|
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index] Re: [Xen-devel] [RFC PATCH 13/19] xen/arm: its: implement hw_irq_controller for LPIs
On Mon, 2 Mar 2015, vijay.kilari@xxxxxxxxx wrote:
> From: Vijaya Kumar K <Vijaya.Kumar@xxxxxxxxxxxxxxxxxx>
>
> This patch implements hw_irq_controller api's required
> to handle LPI's.
>
> Signed-off-by: Vijaya Kumar K <Vijaya.Kumar@xxxxxxxxxxxxxxxxxx>
> ---
> xen/arch/arm/gic-v3-its.c | 104
> ++++++++++++++++++++++++++++++++++++++++++++-
> xen/arch/arm/gic.c | 11 +++++
> xen/include/asm-arm/gic.h | 8 +++-
> 3 files changed, 121 insertions(+), 2 deletions(-)
>
> diff --git a/xen/arch/arm/gic-v3-its.c b/xen/arch/arm/gic-v3-its.c
> index 5d9550f..b2c3320 100644
> --- a/xen/arch/arm/gic-v3-its.c
> +++ b/xen/arch/arm/gic-v3-its.c
> @@ -140,6 +140,9 @@ struct its_cmd_desc {
> typedef struct its_collection *(*its_cmd_builder_t)(struct its_cmd_block *,
> struct its_cmd_desc *);
>
> +static void its_send_movi(struct its_node *its, struct its_collection *col,
> + u32 dev_id, u32 id);
> +
> uint32_t its_get_pta_type(void)
> {
> return pta_type;
> @@ -244,6 +247,103 @@ int its_get_physical_cid(uint32_t *col_id, uint64_t vta)
> return 1;
> }
>
> +static void lpi_set_config(u32 hwirq, u32 id, int enable)
> +{
> + u8 *cfg;
> +
> + cfg = gic_rdists->prop_page + hwirq - NR_GIC_LPI;
> +
> + if ( enable )
> + *cfg |= (1 << 0);
> + else
> + *cfg &= ~(1 << 0);
> +
> + /*
> + * Make the above write visible to the redistributors.
> + * And yes, we're flushing exactly: One. Single. Byte.
> + * Humpf...
> + */
> + if (gic_rdists->flags & RDIST_FLAGS_PROPBASE_NEEDS_FLUSHING)
> + clean_dcache_va_range(cfg, sizeof(*cfg));
> + else
> + dsb(ishst);
> +}
> +
> +static void its_mask_irq(struct irq_desc *d)
> +{
> + u32 id;
> +
> + id = d->irq;
> + set_bit(_IRQ_DISABLED, &d->status);
> + lpi_set_config(d->irq, id, 0);
> +}
> +
> +static void its_unmask_irq(struct irq_desc *d)
> +{
> + u32 id;
> +
> + id = d->irq;
> + clear_bit(_IRQ_DISABLED, &d->status);
> + lpi_set_config(d->irq, id, 1);
> +}
> +
> +static unsigned int its_irq_startup(struct irq_desc *desc)
> +{
> + its_unmask_irq(desc);
> +
> + return 0;
> +}
> +
> +static void its_irq_shutdown(struct irq_desc *desc)
> +{
> + its_mask_irq(desc);
> +}
> +
> +static void its_eoi_irq(struct irq_desc *d)
> +{
> + gic_eoi_irq(d);
> +}
> +
> +static void its_ack_irq(struct irq_desc *desc)
> +{
> + /* No ACK -- reading IAR has done this for us */
> +}
> +
> +static void its_set_affinity(struct irq_desc *d, const cpumask_t *mask_val)
> +{
> + /* XXX: check cpumask_any or cpu_online_map is ok? */
> + cpumask_t online_mask;
> + unsigned int cpu;
> + struct vits_device *its_dev = irq_get_desc_data(d);
> + struct its_collection *target_col;
> + uint32_t id;
> +
> + cpumask_and(&online_mask, mask_val, &cpu_online_map);
> + cpu = cpumask_any(&online_mask);
> + /* Physical collection id */
> + target_col = &its->collections[cpu];
> + /* Physical irq is considered not virq */
> + id = d->irq;
> +
> + its_send_movi(its, target_col, its_dev->dev_id, id);
> +}
> +
> +/* TODO: To implement set_affinity */
> +static const hw_irq_controller gic_guest_its_type = {
> + .typename = "gic-its",
> + .startup = its_irq_startup,
> + .shutdown = its_irq_shutdown,
> + .enable = its_unmask_irq,
> + .disable = its_mask_irq,
> + .ack = its_ack_irq,
> + .end = its_eoi_irq,
> + .set_affinity = its_set_affinity,
> +};
> +
> +static const struct gic_its_hw_operations gic_its_ops = {
> + .gic_guest_lpi_type = &gic_guest_its_type,
> +};
Aside from the one call to its_send_movi, nothing here is ITS specific.
In fact I would just move all this to gic-v3.c as it feels like part of
the GICv3 driver rather than the ITS driver.
> static u64 its_cmd_ptr_to_offset(struct its_node *its,
> struct its_cmd_block *ptr)
> {
> @@ -392,7 +492,7 @@ static void its_send_mapc(struct its_node *its, struct
> its_collection *col,
> its_send_single_command(its, its_build_mapc_cmd, &desc);
> }
>
> -void its_send_movi(struct its_node *its, struct its_collection *col,
> +static void its_send_movi(struct its_node *its, struct its_collection *col,
> u32 dev_id, u32 id)
> {
> struct its_cmd_desc desc;
> @@ -872,6 +972,8 @@ int its_init(struct dt_device_node *node, struct
> rdist_prop *rdists)
>
> its_alloc_lpi_tables();
>
> + register_gic_its_ops(&gic_its_ops);
> +
> return 0;
> }
>
> diff --git a/xen/arch/arm/gic.c b/xen/arch/arm/gic.c
> index 390c8b0..fb77387 100644
> --- a/xen/arch/arm/gic.c
> +++ b/xen/arch/arm/gic.c
> @@ -46,12 +46,18 @@ static DEFINE_PER_CPU(uint64_t, lr_mask);
> static void gic_update_one_lr(struct vcpu *v, int i);
>
> static const struct gic_hw_operations *gic_hw_ops;
> +static const struct gic_its_hw_operations *gic_its_hw_ops;
>
> void register_gic_ops(const struct gic_hw_operations *ops)
> {
> gic_hw_ops = ops;
> }
>
> +void register_gic_its_ops(const struct gic_its_hw_operations *ops)
> +{
> + gic_its_hw_ops = ops;
> +}
Not clear why you need this.
> static void clear_cpu_lr_mask(void)
> {
> this_cpu(lr_mask) = 0ULL;
> @@ -67,6 +73,11 @@ unsigned int gic_number_lines(void)
> return gic_hw_ops->info->nr_lines;
> }
>
> +void gic_eoi_irq(struct irq_desc *d)
> +{
> + gic_hw_ops->eoi_irq(d);
> +}
> +
> void gic_save_state(struct vcpu *v)
> {
> ASSERT(!local_irq_is_enabled());
> diff --git a/xen/include/asm-arm/gic.h b/xen/include/asm-arm/gic.h
> index 681d75c..eac738f 100644
> --- a/xen/include/asm-arm/gic.h
> +++ b/xen/include/asm-arm/gic.h
> @@ -217,7 +217,7 @@ enum gic_version {
> };
>
> extern enum gic_version gic_hw_version(void);
> -
> +extern void gic_eoi_irq(struct irq_desc *desc);
> /* Program the GIC to route an interrupt */
> extern void gic_route_irq_to_xen(struct irq_desc *desc, const cpumask_t
> *cpu_mask,
> unsigned int priority);
> @@ -291,6 +291,11 @@ struct gic_info {
> unsigned int maintenance_irq;
> };
>
> +struct gic_its_hw_operations {
> + /* LPI hw_irq_controller to enable/disable/eoi host irq */
> + hw_irq_controller *gic_guest_lpi_type;
> +};
> +
> struct gic_hw_operations {
> /* Hold GIC HW information */
> const struct gic_info *info;
> @@ -346,6 +351,7 @@ struct gic_hw_operations {
> };
>
> void register_gic_ops(const struct gic_hw_operations *ops);
> +void register_gic_its_ops(const struct gic_its_hw_operations *ops);
> int gic_make_node(const struct domain *d,const struct dt_device_node *node,
> void *fdt);
>
> --
> 1.7.9.5
>
_______________________________________________
Xen-devel mailing list
Xen-devel@xxxxxxxxxxxxx
http://lists.xen.org/xen-devel
|
![]() |
Lists.xenproject.org is hosted with RackSpace, monitoring our |