On 8/15/07, Blue Swirl <[EMAIL PROTECTED]> wrote:
> On 8/15/07, Paul Brook <[EMAIL PROTECTED]> wrote:
> > On Wednesday 15 August 2007, Blue Swirl wrote:
> > > On 8/15/07, Paul Brook <[EMAIL PROTECTED]> wrote:
> > > > On Wednesday 15 August 2007, Paul Brook wrote:
> > > > > > Okay, more explaining. This is the case where I'd want to use the
> > > > > > signal: DMA controller ("upstream") can reset the slave device (ESP
> > > > > > or Lance). DMA controller is created first and I also want to
> > > > > > allocate reset signals at that point. Later when ESP is created, it
> > > > > > should be possible to put ESP reset function and opaque data to the
> > > > > > signal given but this is not possible with current API. Currently 
> > > > > > the
> > > > > > DMA data would be passed to qemu_allocate_irqs.
> > > > >
> > > > > Ah, I see. The problem here is that you've got a cyclic dependency. 
> > > > > For
> > > > > DMA operations the ESP is in charge, so it makes sense to create the
> > > > > subservient DMA device first. For the reset signals the DMA controller
> > > > > is in charge so ideally you create the ESP device first. Because the
> > > > > DMA interface is most complicated, it's probably takes precedence.
> > > > >
> > > > > I think you need to modify or use sparc32_dma_set_reset_data to take a
> > > > > qemu_irq rather than a callback and opaque argument. Alternatively you
> > > > > can move things around a bit and have the sun4m code do something
> > > > > similar. i.e. the ESP and lance devices return the reset lines, then
> > > > > the sun4m code pokes into the DMA device state.
> > > >
> > > > Oh, or you can pass a pointer to a qemu_irq from the DMA to the ESP and
> > > > have the ESP poke its reset object in there that way.
> > >
> > > That's what I had in mind. Should I just extend the API for example with
> > > /* Change the callback function and/or data */
> > > void qemu_irq_change(qemu_irq irq, qemu_irq_handler handler,
> > >                      void *opaque);
> >
> > I'm not so keen on that. It breaks the nice property of having the IRQ lines
> > owned by the receiver. I was thinking something like:
> >
> > struct DMAState {
> >   qemu_irq device_reset;
> > };
> > qemu_irq *dma_init()
> > {
> >   struct DMAState * d = malloc();
> >   return &d->device_reset;
> > }
> >
> > void esp_init(qemu_irq *resetp)
> > {
> >   ESPState *d = malloc();
> >   *reset = *qemu_irq_alloc(d, 1);
> > }
> >
> > void sun4m_init()
> > {
> >   qemu_irq *p = dma_init();
> >   esp_init(p);
> > }
>
> Yes, that would work. I wasn't too happy about the change function either.
>

I implemented reset that way with qemu_irq, the result survives some
quick tests.
Index: qemu/hw/esp.c
===================================================================
--- qemu.orig/hw/esp.c	2007-08-15 18:58:02.000000000 +0000
+++ qemu/hw/esp.c	2007-08-15 19:22:38.000000000 +0000
@@ -344,6 +344,11 @@
     s->do_cmd = 0;
 }
 
+static void parent_esp_reset(void *opaque, int irq, int level)
+{
+    esp_reset(opaque);
+}
+
 static uint32_t esp_mem_readb(void *opaque, target_phys_addr_t addr)
 {
     ESPState *s = opaque;
@@ -569,7 +574,7 @@
 }
 
 void *esp_init(BlockDriverState **bd, target_phys_addr_t espaddr,
-               void *dma_opaque, qemu_irq irq, qemu_dma *parent_dma)
+               qemu_irq irq, qemu_dma *parent_dma, qemu_irq *reset)
 {
     ESPState *s;
     int esp_io_memory;
@@ -581,7 +586,6 @@
     s->bd = bd;
     s->irq = irq;
     s->parent_dma = parent_dma;
-    sparc32_dma_set_reset_data(dma_opaque, esp_reset, s);
 
     esp_io_memory = cpu_register_io_memory(0, esp_mem_read, esp_mem_write, s);
     cpu_register_physical_memory(espaddr, ESP_SIZE, esp_io_memory);
@@ -591,5 +595,7 @@
     register_savevm("esp", espaddr, 3, esp_save, esp_load, s);
     qemu_register_reset(esp_reset, s);
 
+    *reset = *qemu_allocate_irqs(parent_esp_reset, s, 1);
+
     return s;
 }
Index: qemu/hw/sun4m.c
===================================================================
--- qemu.orig/hw/sun4m.c	2007-08-15 18:58:02.000000000 +0000
+++ qemu/hw/sun4m.c	2007-08-15 19:05:57.000000000 +0000
@@ -322,6 +322,7 @@
     qemu_irq *cpu_irqs[MAX_CPUS], *slavio_irq, *slavio_cpu_irq,
         *espdma_irq, *ledma_irq;
     qemu_dma *physical_dma, *dvma, *esp_dvma, *le_dvma;
+    qemu_irq *esp_reset, *le_reset;
 
     /* init CPUs */
     sparc_find_by_name(cpu_model, &def);
@@ -362,11 +363,11 @@
                                        hwdef->clock_irq);
 
     espdma = sparc32_dma_init(hwdef->dma_base, slavio_irq[hwdef->esp_irq],
-                              &espdma_irq, dvma, &esp_dvma, 1);
+                              &espdma_irq, dvma, &esp_dvma, 1, &esp_reset);
 
     ledma = sparc32_dma_init(hwdef->dma_base + 16ULL,
                              slavio_irq[hwdef->le_irq], &ledma_irq, dvma,
-                             &le_dvma, 0);
+                             &le_dvma, 0, &le_reset);
 
     if (graphic_depth != 8 && graphic_depth != 24) {
         fprintf(stderr, "qemu: Unsupported depth: %d\n", graphic_depth);
@@ -377,7 +378,7 @@
 
     if (nd_table[0].model == NULL
         || strcmp(nd_table[0].model, "lance") == 0) {
-        lance_init(&nd_table[0], hwdef->le_base, ledma, *ledma_irq, le_dvma);
+        lance_init(&nd_table[0], hwdef->le_base, *ledma_irq, le_dvma, le_reset);
     } else if (strcmp(nd_table[0].model, "?") == 0) {
         fprintf(stderr, "qemu: Supported NICs: lance\n");
         exit (1);
@@ -402,8 +403,8 @@
                        serial_hds[1], serial_hds[0]);
     fdctrl_init(slavio_irq[hwdef->fd_irq], 0, 1, hwdef->fd_base, fd_table);
 
-    main_esp = esp_init(bs_table, hwdef->esp_base, espdma, *espdma_irq,
-                        esp_dvma);
+    main_esp = esp_init(bs_table, hwdef->esp_base, *espdma_irq, esp_dvma,
+                        esp_reset);
 
     for (i = 0; i < MAX_DISKS; i++) {
         if (bs_table[i]) {
Index: qemu/vl.h
===================================================================
--- qemu.orig/vl.h	2007-08-15 18:58:02.000000000 +0000
+++ qemu/vl.h	2007-08-15 19:05:57.000000000 +0000
@@ -1097,8 +1097,8 @@
 /* pcnet.c */
 
 void pci_pcnet_init(PCIBus *bus, NICInfo *nd, int devfn);
-void lance_init(NICInfo *nd, target_phys_addr_t leaddr, void *dma_opaque,
-                qemu_irq irq, qemu_dma *parent_dma);
+void lance_init(NICInfo *nd, target_phys_addr_t leaddr, qemu_irq irq,
+                qemu_dma *parent_dma, qemu_irq *reset);
 
 /* vmmouse.c */
 void *vmmouse_init(void *m);
@@ -1303,14 +1303,12 @@
 /* esp.c */
 void esp_scsi_attach(void *opaque, BlockDriverState *bd, int id);
 void *esp_init(BlockDriverState **bd, target_phys_addr_t espaddr,
-               void *dma_opaque, qemu_irq irq, qemu_dma *parent_dma);
+               qemu_irq irq, qemu_dma *parent_dma, qemu_irq *reset);
 
 /* sparc32_dma.c */
 void *sparc32_dma_init(target_phys_addr_t daddr, qemu_irq parent_irq,
                        qemu_irq **dev_irq, qemu_dma *parent_dma,
-                       qemu_dma **dev_dma, int is_espdma);
-void sparc32_dma_set_reset_data(void *opaque, void (*dev_reset)(void *opaque),
-                                void *dev_opaque);
+                       qemu_dma **dev_dma, int is_espdma, qemu_irq **reset);
 
 /* cs4231.c */
 void cs_init(target_phys_addr_t base, int irq, void *intctl);
Index: qemu/hw/pcnet.c
===================================================================
--- qemu.orig/hw/pcnet.c	2007-08-15 18:58:02.000000000 +0000
+++ qemu/hw/pcnet.c	2007-08-15 19:05:57.000000000 +0000
@@ -2052,6 +2052,11 @@
     }
 }
 
+static void parent_lance_reset(void *opaque, int irq, int level)
+{
+    pcnet_h_reset(opaque);
+}
+
 static void lance_mem_writew(void *opaque, target_phys_addr_t addr,
                              uint32_t val)
 {
@@ -2087,8 +2092,8 @@
     lance_mem_writew,
 };
 
-void lance_init(NICInfo *nd, target_phys_addr_t leaddr, void *dma_opaque,
-                qemu_irq irq, qemu_dma *parent_dma)
+void lance_init(NICInfo *nd, target_phys_addr_t leaddr, qemu_irq irq,
+                qemu_dma *parent_dma, qemu_irq *reset)
 {
     PCNetState *d;
     int lance_io_memory;
@@ -2101,7 +2106,7 @@
         cpu_register_io_memory(0, lance_mem_read, lance_mem_write, d);
 
     d->dma_opaque = parent_dma;
-    sparc32_dma_set_reset_data(dma_opaque, pcnet_h_reset, d);
+    *reset = *qemu_allocate_irqs(parent_lance_reset, d, 1);
 
     cpu_register_physical_memory(leaddr, 4, lance_io_memory);
 
Index: qemu/hw/sparc32_dma.c
===================================================================
--- qemu.orig/hw/sparc32_dma.c	2007-08-15 18:58:02.000000000 +0000
+++ qemu/hw/sparc32_dma.c	2007-08-15 19:25:46.000000000 +0000
@@ -58,9 +58,8 @@
 struct DMAState {
     uint32_t dmaregs[DMA_REGS];
     qemu_irq irq;
-    void *dev_opaque;
-    void (*dev_reset)(void *dev_opaque);
     qemu_irq *pic;
+    qemu_irq dev_reset;
     qemu_dma *dma;
 };
 
@@ -132,7 +131,7 @@
             qemu_irq_lower(s->irq);
         }
         if (val & DMA_RESET) {
-            s->dev_reset(s->dev_opaque);
+            qemu_irq_raise(s->dev_reset);
         } else if (val & DMA_DRAIN_FIFO) {
             val &= ~DMA_DRAIN_FIFO;
         } else if (val == 0)
@@ -193,7 +192,7 @@
 
 void *sparc32_dma_init(target_phys_addr_t daddr, qemu_irq parent_irq,
                        qemu_irq **dev_irq, qemu_dma *parent_dma,
-                       qemu_dma **dev_dma, int is_espdma)
+                       qemu_dma **dev_dma, int is_espdma, qemu_irq **reset)
 {
     DMAState *s;
     int dma_io_memory;
@@ -217,14 +216,7 @@
     else
         *dev_dma = qemu_init_dma(ledma_memory_rw, s);
 
-    return s;
-}
-
-void sparc32_dma_set_reset_data(void *opaque, void (*dev_reset)(void *opaque),
-                                void *dev_opaque)
-{
-    DMAState *s = opaque;
+    *reset = &s->dev_reset;
 
-    s->dev_reset = dev_reset;
-    s->dev_opaque = dev_opaque;
+    return s;
 }

Reply via email to