Infdf029762fwe factored out the handling of reading and writing DMA descriptors from guest memory. Unfortunately we accidentally made the descriptor-read read the descriptor into the address of the buffer rather than into the buffer, because we didn't notice we needed to update the arguments to the dma_memory_read() call. Before the refactoring, "&desc" is the address of a local struct DPDMADescriptor variable in xlnx_dpdma_start_operation(), which is the correct target for the guest-memory-read. But after the refactoring 'desc' is the "DPDMADescriptor *desc" argument to the new function, and so it is already an address. This bug is an overrun of a stack variable, since a pointer is at most 8 bytes long and we try to read 64 bytes, as well as being incorrect behaviour. Pass 'desc' rather than '&desc' as the dma_memory_read() argument to fix this. (The same bug is not present in xlnx_dpdma_write_descriptor(), because there we are writing the descriptor from a local struct variable "DPDMADescriptor tmp_desc" and so passing &tmp_desc to dma_memory_write() is correct.) Spotted by Coverity: CID 1546649 Fixes:fdf029762f("xlnx_dpdma: fix descriptor endianness bug") Signed-off-by: Peter Maydell <peter.maydell@linaro.org> Reviewed-by: Edgar E. Iglesias <edgar.iglesias@amd.com> Reviewed-by: Philippe Mathieu-Daudé <philmd@linaro.org> Message-ID: <20240531124628.476938-1-peter.maydell@linaro.org> Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org>
		
			
				
	
	
		
			853 lines
		
	
	
		
			29 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			853 lines
		
	
	
		
			29 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * xlnx_dpdma.c
 | 
						|
 *
 | 
						|
 *  Copyright (C) 2015 : GreenSocs Ltd
 | 
						|
 *      http://www.greensocs.com/ , email: info@greensocs.com
 | 
						|
 *
 | 
						|
 *  Developed by :
 | 
						|
 *  Frederic Konrad   <fred.konrad@greensocs.com>
 | 
						|
 *
 | 
						|
 * This program is free software; you can redistribute it and/or modify
 | 
						|
 * it under the terms of the GNU General Public License as published by
 | 
						|
 * the Free Software Foundation, either version 2 of the License, or
 | 
						|
 * (at your option) any later version.
 | 
						|
 *
 | 
						|
 * This program is distributed in the hope that it will be useful,
 | 
						|
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
 * GNU General Public License for more details.
 | 
						|
 *
 | 
						|
 * You should have received a copy of the GNU General Public License along
 | 
						|
 * with this program; if not, see <http://www.gnu.org/licenses/>.
 | 
						|
 *
 | 
						|
 */
 | 
						|
 | 
						|
#include "qemu/osdep.h"
 | 
						|
#include "qemu/cutils.h"
 | 
						|
#include "qemu/log.h"
 | 
						|
#include "qemu/module.h"
 | 
						|
#include "hw/dma/xlnx_dpdma.h"
 | 
						|
#include "hw/irq.h"
 | 
						|
#include "migration/vmstate.h"
 | 
						|
 | 
						|
#ifndef DEBUG_DPDMA
 | 
						|
#define DEBUG_DPDMA 0
 | 
						|
#endif
 | 
						|
 | 
						|
#define DPRINTF(fmt, ...) do {                                                 \
 | 
						|
    if (DEBUG_DPDMA) {                                                         \
 | 
						|
        qemu_log("xlnx_dpdma: " fmt , ## __VA_ARGS__);                         \
 | 
						|
    }                                                                          \
 | 
						|
} while (0)
 | 
						|
 | 
						|
/*
 | 
						|
 * Registers offset for DPDMA.
 | 
						|
 */
 | 
						|
#define DPDMA_ERR_CTRL                        (0x0000)
 | 
						|
#define DPDMA_ISR                             (0x0004 >> 2)
 | 
						|
#define DPDMA_IMR                             (0x0008 >> 2)
 | 
						|
#define DPDMA_IEN                             (0x000C >> 2)
 | 
						|
#define DPDMA_IDS                             (0x0010 >> 2)
 | 
						|
#define DPDMA_EISR                            (0x0014 >> 2)
 | 
						|
#define DPDMA_EIMR                            (0x0018 >> 2)
 | 
						|
#define DPDMA_EIEN                            (0x001C >> 2)
 | 
						|
#define DPDMA_EIDS                            (0x0020 >> 2)
 | 
						|
#define DPDMA_CNTL                            (0x0100 >> 2)
 | 
						|
 | 
						|
#define DPDMA_GBL                             (0x0104 >> 2)
 | 
						|
#define DPDMA_GBL_TRG_CH(n)                   (1 << n)
 | 
						|
#define DPDMA_GBL_RTRG_CH(n)                  (1 << 6 << n)
 | 
						|
 | 
						|
#define DPDMA_ALC0_CNTL                       (0x0108 >> 2)
 | 
						|
#define DPDMA_ALC0_STATUS                     (0x010C >> 2)
 | 
						|
#define DPDMA_ALC0_MAX                        (0x0110 >> 2)
 | 
						|
#define DPDMA_ALC0_MIN                        (0x0114 >> 2)
 | 
						|
#define DPDMA_ALC0_ACC                        (0x0118 >> 2)
 | 
						|
#define DPDMA_ALC0_ACC_TRAN                   (0x011C >> 2)
 | 
						|
#define DPDMA_ALC1_CNTL                       (0x0120 >> 2)
 | 
						|
#define DPDMA_ALC1_STATUS                     (0x0124 >> 2)
 | 
						|
#define DPDMA_ALC1_MAX                        (0x0128 >> 2)
 | 
						|
#define DPDMA_ALC1_MIN                        (0x012C >> 2)
 | 
						|
#define DPDMA_ALC1_ACC                        (0x0130 >> 2)
 | 
						|
#define DPDMA_ALC1_ACC_TRAN                   (0x0134 >> 2)
 | 
						|
 | 
						|
#define DPDMA_DSCR_STRT_ADDRE_CH(n)           ((0x0200 + n * 0x100) >> 2)
 | 
						|
#define DPDMA_DSCR_STRT_ADDR_CH(n)            ((0x0204 + n * 0x100) >> 2)
 | 
						|
#define DPDMA_DSCR_NEXT_ADDRE_CH(n)           ((0x0208 + n * 0x100) >> 2)
 | 
						|
#define DPDMA_DSCR_NEXT_ADDR_CH(n)            ((0x020C + n * 0x100) >> 2)
 | 
						|
#define DPDMA_PYLD_CUR_ADDRE_CH(n)            ((0x0210 + n * 0x100) >> 2)
 | 
						|
#define DPDMA_PYLD_CUR_ADDR_CH(n)             ((0x0214 + n * 0x100) >> 2)
 | 
						|
 | 
						|
#define DPDMA_CNTL_CH(n)                      ((0x0218 + n * 0x100) >> 2)
 | 
						|
#define DPDMA_CNTL_CH_EN                      (1)
 | 
						|
#define DPDMA_CNTL_CH_PAUSED                  (1 << 1)
 | 
						|
 | 
						|
#define DPDMA_STATUS_CH(n)                    ((0x021C + n * 0x100) >> 2)
 | 
						|
#define DPDMA_STATUS_BURST_TYPE               (1 << 4)
 | 
						|
#define DPDMA_STATUS_MODE                     (1 << 5)
 | 
						|
#define DPDMA_STATUS_EN_CRC                   (1 << 6)
 | 
						|
#define DPDMA_STATUS_LAST_DSCR                (1 << 7)
 | 
						|
#define DPDMA_STATUS_LDSCR_FRAME              (1 << 8)
 | 
						|
#define DPDMA_STATUS_IGNR_DONE                (1 << 9)
 | 
						|
#define DPDMA_STATUS_DSCR_DONE                (1 << 10)
 | 
						|
#define DPDMA_STATUS_EN_DSCR_UP               (1 << 11)
 | 
						|
#define DPDMA_STATUS_EN_DSCR_INTR             (1 << 12)
 | 
						|
#define DPDMA_STATUS_PREAMBLE_OFF             (13)
 | 
						|
 | 
						|
#define DPDMA_VDO_CH(n)                       ((0x0220 + n * 0x100) >> 2)
 | 
						|
#define DPDMA_PYLD_SZ_CH(n)                   ((0x0224 + n * 0x100) >> 2)
 | 
						|
#define DPDMA_DSCR_ID_CH(n)                   ((0x0228 + n * 0x100) >> 2)
 | 
						|
 | 
						|
/*
 | 
						|
 * Descriptor control field.
 | 
						|
 */
 | 
						|
#define CONTROL_PREAMBLE_VALUE                0xA5
 | 
						|
 | 
						|
#define DSCR_CTRL_PREAMBLE                    0xFF
 | 
						|
#define DSCR_CTRL_EN_DSCR_DONE_INTR           (1 << 8)
 | 
						|
#define DSCR_CTRL_EN_DSCR_UPDATE              (1 << 9)
 | 
						|
#define DSCR_CTRL_IGNORE_DONE                 (1 << 10)
 | 
						|
#define DSCR_CTRL_AXI_BURST_TYPE              (1 << 11)
 | 
						|
#define DSCR_CTRL_AXCACHE                     (0x0F << 12)
 | 
						|
#define DSCR_CTRL_AXPROT                      (0x2 << 16)
 | 
						|
#define DSCR_CTRL_DESCRIPTOR_MODE             (1 << 18)
 | 
						|
#define DSCR_CTRL_LAST_DESCRIPTOR             (1 << 19)
 | 
						|
#define DSCR_CTRL_ENABLE_CRC                  (1 << 20)
 | 
						|
#define DSCR_CTRL_LAST_DESCRIPTOR_OF_FRAME    (1 << 21)
 | 
						|
 | 
						|
/*
 | 
						|
 * Descriptor timestamp field.
 | 
						|
 */
 | 
						|
#define STATUS_DONE                           (1 << 31)
 | 
						|
 | 
						|
#define DPDMA_FRAG_MAX_SZ                     (4096)
 | 
						|
 | 
						|
enum DPDMABurstType {
 | 
						|
    DPDMA_INCR = 0,
 | 
						|
    DPDMA_FIXED = 1
 | 
						|
};
 | 
						|
 | 
						|
enum DPDMAMode {
 | 
						|
    DPDMA_CONTIGOUS = 0,
 | 
						|
    DPDMA_FRAGMENTED = 1
 | 
						|
};
 | 
						|
 | 
						|
struct DPDMADescriptor {
 | 
						|
    uint32_t control;
 | 
						|
    uint32_t descriptor_id;
 | 
						|
    /* transfer size in byte. */
 | 
						|
    uint32_t xfer_size;
 | 
						|
    uint32_t line_size_stride;
 | 
						|
    uint32_t timestamp_lsb;
 | 
						|
    uint32_t timestamp_msb;
 | 
						|
    /* contains extension for both descriptor and source. */
 | 
						|
    uint32_t address_extension;
 | 
						|
    uint32_t next_descriptor;
 | 
						|
    uint32_t source_address;
 | 
						|
    uint32_t address_extension_23;
 | 
						|
    uint32_t address_extension_45;
 | 
						|
    uint32_t source_address2;
 | 
						|
    uint32_t source_address3;
 | 
						|
    uint32_t source_address4;
 | 
						|
    uint32_t source_address5;
 | 
						|
    uint32_t crc;
 | 
						|
};
 | 
						|
 | 
						|
typedef enum DPDMABurstType DPDMABurstType;
 | 
						|
typedef enum DPDMAMode DPDMAMode;
 | 
						|
typedef struct DPDMADescriptor DPDMADescriptor;
 | 
						|
 | 
						|
static bool xlnx_dpdma_desc_is_last(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    return ((desc->control & DSCR_CTRL_LAST_DESCRIPTOR) != 0);
 | 
						|
}
 | 
						|
 | 
						|
static bool xlnx_dpdma_desc_is_last_of_frame(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    return ((desc->control & DSCR_CTRL_LAST_DESCRIPTOR_OF_FRAME) != 0);
 | 
						|
}
 | 
						|
 | 
						|
static uint64_t xlnx_dpdma_desc_get_source_address(DPDMADescriptor *desc,
 | 
						|
                                                     uint8_t frag)
 | 
						|
{
 | 
						|
    uint64_t addr = 0;
 | 
						|
    assert(frag < 5);
 | 
						|
 | 
						|
    switch (frag) {
 | 
						|
    case 0:
 | 
						|
        addr = (uint64_t)desc->source_address
 | 
						|
            + (extract64(desc->address_extension, 16, 16) << 32);
 | 
						|
        break;
 | 
						|
    case 1:
 | 
						|
        addr = (uint64_t)desc->source_address2
 | 
						|
            + (extract64(desc->address_extension_23, 0, 16) << 32);
 | 
						|
        break;
 | 
						|
    case 2:
 | 
						|
        addr = (uint64_t)desc->source_address3
 | 
						|
            + (extract64(desc->address_extension_23, 16, 16) << 32);
 | 
						|
        break;
 | 
						|
    case 3:
 | 
						|
        addr = (uint64_t)desc->source_address4
 | 
						|
            + (extract64(desc->address_extension_45, 0, 16) << 32);
 | 
						|
        break;
 | 
						|
    case 4:
 | 
						|
        addr = (uint64_t)desc->source_address5
 | 
						|
            + (extract64(desc->address_extension_45, 16, 16) << 32);
 | 
						|
        break;
 | 
						|
    default:
 | 
						|
        addr = 0;
 | 
						|
        break;
 | 
						|
    }
 | 
						|
 | 
						|
    return addr;
 | 
						|
}
 | 
						|
 | 
						|
static uint32_t xlnx_dpdma_desc_get_transfer_size(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    return desc->xfer_size;
 | 
						|
}
 | 
						|
 | 
						|
static uint32_t xlnx_dpdma_desc_get_line_size(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    return extract32(desc->line_size_stride, 0, 18);
 | 
						|
}
 | 
						|
 | 
						|
static uint32_t xlnx_dpdma_desc_get_line_stride(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    return extract32(desc->line_size_stride, 18, 14) * 16;
 | 
						|
}
 | 
						|
 | 
						|
static inline bool xlnx_dpdma_desc_crc_enabled(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    return (desc->control & DSCR_CTRL_ENABLE_CRC) != 0;
 | 
						|
}
 | 
						|
 | 
						|
static inline bool xlnx_dpdma_desc_check_crc(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    uint32_t *p = (uint32_t *)desc;
 | 
						|
    uint32_t crc = 0;
 | 
						|
    uint8_t i;
 | 
						|
 | 
						|
    /*
 | 
						|
     * CRC is calculated on the whole descriptor except the last 32bits word
 | 
						|
     * using 32bits addition.
 | 
						|
     */
 | 
						|
    for (i = 0; i < 15; i++) {
 | 
						|
        crc += p[i];
 | 
						|
    }
 | 
						|
 | 
						|
    return crc == desc->crc;
 | 
						|
}
 | 
						|
 | 
						|
static inline bool xlnx_dpdma_desc_completion_interrupt(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    return (desc->control & DSCR_CTRL_EN_DSCR_DONE_INTR) != 0;
 | 
						|
}
 | 
						|
 | 
						|
static inline bool xlnx_dpdma_desc_is_valid(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    return (desc->control & DSCR_CTRL_PREAMBLE) == CONTROL_PREAMBLE_VALUE;
 | 
						|
}
 | 
						|
 | 
						|
static inline bool xlnx_dpdma_desc_is_contiguous(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    return (desc->control & DSCR_CTRL_DESCRIPTOR_MODE) == 0;
 | 
						|
}
 | 
						|
 | 
						|
static inline bool xlnx_dpdma_desc_update_enabled(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    return (desc->control & DSCR_CTRL_EN_DSCR_UPDATE) != 0;
 | 
						|
}
 | 
						|
 | 
						|
static inline void xlnx_dpdma_desc_set_done(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    desc->timestamp_msb |= STATUS_DONE;
 | 
						|
}
 | 
						|
 | 
						|
static inline bool xlnx_dpdma_desc_is_already_done(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    return (desc->timestamp_msb & STATUS_DONE) != 0;
 | 
						|
}
 | 
						|
 | 
						|
static inline bool xlnx_dpdma_desc_ignore_done_bit(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    return (desc->control & DSCR_CTRL_IGNORE_DONE) != 0;
 | 
						|
}
 | 
						|
 | 
						|
static const VMStateDescription vmstate_xlnx_dpdma = {
 | 
						|
    .name = TYPE_XLNX_DPDMA,
 | 
						|
    .version_id = 1,
 | 
						|
    .fields = (const VMStateField[]) {
 | 
						|
        VMSTATE_UINT32_ARRAY(registers, XlnxDPDMAState,
 | 
						|
                             XLNX_DPDMA_REG_ARRAY_SIZE),
 | 
						|
        VMSTATE_BOOL_ARRAY(operation_finished, XlnxDPDMAState, 6),
 | 
						|
        VMSTATE_END_OF_LIST()
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
static void xlnx_dpdma_update_irq(XlnxDPDMAState *s)
 | 
						|
{
 | 
						|
    bool flags;
 | 
						|
 | 
						|
    flags = ((s->registers[DPDMA_ISR] & (~s->registers[DPDMA_IMR]))
 | 
						|
          || (s->registers[DPDMA_EISR] & (~s->registers[DPDMA_EIMR])));
 | 
						|
    qemu_set_irq(s->irq, flags);
 | 
						|
}
 | 
						|
 | 
						|
static uint64_t xlnx_dpdma_descriptor_start_address(XlnxDPDMAState *s,
 | 
						|
                                                      uint8_t channel)
 | 
						|
{
 | 
						|
    return (s->registers[DPDMA_DSCR_STRT_ADDRE_CH(channel)] << 16)
 | 
						|
          + s->registers[DPDMA_DSCR_STRT_ADDR_CH(channel)];
 | 
						|
}
 | 
						|
 | 
						|
static uint64_t xlnx_dpdma_descriptor_next_address(XlnxDPDMAState *s,
 | 
						|
                                                     uint8_t channel)
 | 
						|
{
 | 
						|
    return ((uint64_t)s->registers[DPDMA_DSCR_NEXT_ADDRE_CH(channel)] << 32)
 | 
						|
           + s->registers[DPDMA_DSCR_NEXT_ADDR_CH(channel)];
 | 
						|
}
 | 
						|
 | 
						|
static bool xlnx_dpdma_is_channel_enabled(XlnxDPDMAState *s,
 | 
						|
                                            uint8_t channel)
 | 
						|
{
 | 
						|
    return (s->registers[DPDMA_CNTL_CH(channel)] & DPDMA_CNTL_CH_EN) != 0;
 | 
						|
}
 | 
						|
 | 
						|
static bool xlnx_dpdma_is_channel_paused(XlnxDPDMAState *s,
 | 
						|
                                           uint8_t channel)
 | 
						|
{
 | 
						|
    return (s->registers[DPDMA_CNTL_CH(channel)] & DPDMA_CNTL_CH_PAUSED) != 0;
 | 
						|
}
 | 
						|
 | 
						|
static inline bool xlnx_dpdma_is_channel_retriggered(XlnxDPDMAState *s,
 | 
						|
                                                       uint8_t channel)
 | 
						|
{
 | 
						|
    /* Clear the retriggered bit after reading it. */
 | 
						|
    bool channel_is_retriggered = s->registers[DPDMA_GBL]
 | 
						|
                                & DPDMA_GBL_RTRG_CH(channel);
 | 
						|
    s->registers[DPDMA_GBL] &= ~DPDMA_GBL_RTRG_CH(channel);
 | 
						|
    return channel_is_retriggered;
 | 
						|
}
 | 
						|
 | 
						|
static inline bool xlnx_dpdma_is_channel_triggered(XlnxDPDMAState *s,
 | 
						|
                                                     uint8_t channel)
 | 
						|
{
 | 
						|
    return s->registers[DPDMA_GBL] & DPDMA_GBL_TRG_CH(channel);
 | 
						|
}
 | 
						|
 | 
						|
static void xlnx_dpdma_update_desc_info(XlnxDPDMAState *s, uint8_t channel,
 | 
						|
                                          DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    s->registers[DPDMA_DSCR_NEXT_ADDRE_CH(channel)] =
 | 
						|
                                extract32(desc->address_extension, 0, 16);
 | 
						|
    s->registers[DPDMA_DSCR_NEXT_ADDR_CH(channel)] = desc->next_descriptor;
 | 
						|
    s->registers[DPDMA_PYLD_CUR_ADDRE_CH(channel)] =
 | 
						|
                                extract32(desc->address_extension, 16, 16);
 | 
						|
    s->registers[DPDMA_PYLD_CUR_ADDR_CH(channel)] = desc->source_address;
 | 
						|
    s->registers[DPDMA_VDO_CH(channel)] =
 | 
						|
                                extract32(desc->line_size_stride, 18, 14)
 | 
						|
                                + (extract32(desc->line_size_stride, 0, 18)
 | 
						|
                                  << 14);
 | 
						|
    s->registers[DPDMA_PYLD_SZ_CH(channel)] = desc->xfer_size;
 | 
						|
    s->registers[DPDMA_DSCR_ID_CH(channel)] = desc->descriptor_id;
 | 
						|
 | 
						|
    /* Compute the status register with the descriptor information. */
 | 
						|
    s->registers[DPDMA_STATUS_CH(channel)] =
 | 
						|
                                extract32(desc->control, 0, 8) << 13;
 | 
						|
    if ((desc->control & DSCR_CTRL_EN_DSCR_DONE_INTR) != 0) {
 | 
						|
        s->registers[DPDMA_STATUS_CH(channel)] |= DPDMA_STATUS_EN_DSCR_INTR;
 | 
						|
    }
 | 
						|
    if ((desc->control & DSCR_CTRL_EN_DSCR_UPDATE) != 0) {
 | 
						|
        s->registers[DPDMA_STATUS_CH(channel)] |= DPDMA_STATUS_EN_DSCR_UP;
 | 
						|
    }
 | 
						|
    if ((desc->timestamp_msb & STATUS_DONE) != 0) {
 | 
						|
        s->registers[DPDMA_STATUS_CH(channel)] |= DPDMA_STATUS_DSCR_DONE;
 | 
						|
    }
 | 
						|
    if ((desc->control & DSCR_CTRL_IGNORE_DONE) != 0) {
 | 
						|
        s->registers[DPDMA_STATUS_CH(channel)] |= DPDMA_STATUS_IGNR_DONE;
 | 
						|
    }
 | 
						|
    if ((desc->control & DSCR_CTRL_LAST_DESCRIPTOR_OF_FRAME) != 0) {
 | 
						|
        s->registers[DPDMA_STATUS_CH(channel)] |= DPDMA_STATUS_LDSCR_FRAME;
 | 
						|
    }
 | 
						|
    if ((desc->control & DSCR_CTRL_LAST_DESCRIPTOR) != 0) {
 | 
						|
        s->registers[DPDMA_STATUS_CH(channel)] |= DPDMA_STATUS_LAST_DSCR;
 | 
						|
    }
 | 
						|
    if ((desc->control & DSCR_CTRL_ENABLE_CRC) != 0) {
 | 
						|
        s->registers[DPDMA_STATUS_CH(channel)] |= DPDMA_STATUS_EN_CRC;
 | 
						|
    }
 | 
						|
    if ((desc->control & DSCR_CTRL_DESCRIPTOR_MODE) != 0) {
 | 
						|
        s->registers[DPDMA_STATUS_CH(channel)] |= DPDMA_STATUS_MODE;
 | 
						|
    }
 | 
						|
    if ((desc->control & DSCR_CTRL_AXI_BURST_TYPE) != 0) {
 | 
						|
        s->registers[DPDMA_STATUS_CH(channel)] |= DPDMA_STATUS_BURST_TYPE;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void xlnx_dpdma_dump_descriptor(DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    if (DEBUG_DPDMA) {
 | 
						|
        qemu_log("DUMP DESCRIPTOR:\n");
 | 
						|
        qemu_hexdump(stdout, "", desc, sizeof(DPDMADescriptor));
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static uint64_t xlnx_dpdma_read(void *opaque, hwaddr offset,
 | 
						|
                                unsigned size)
 | 
						|
{
 | 
						|
    XlnxDPDMAState *s = XLNX_DPDMA(opaque);
 | 
						|
 | 
						|
    DPRINTF("read @%" HWADDR_PRIx "\n", offset);
 | 
						|
    offset = offset >> 2;
 | 
						|
 | 
						|
    switch (offset) {
 | 
						|
    /*
 | 
						|
     * Trying to read a write only register.
 | 
						|
     */
 | 
						|
    case DPDMA_GBL:
 | 
						|
        return 0;
 | 
						|
    default:
 | 
						|
        assert(offset <= (0xFFC >> 2));
 | 
						|
        return s->registers[offset];
 | 
						|
    }
 | 
						|
    return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void xlnx_dpdma_write(void *opaque, hwaddr offset,
 | 
						|
                               uint64_t value, unsigned size)
 | 
						|
{
 | 
						|
    XlnxDPDMAState *s = XLNX_DPDMA(opaque);
 | 
						|
 | 
						|
    DPRINTF("write @%" HWADDR_PRIx " = %" PRIx64 "\n", offset, value);
 | 
						|
    offset = offset >> 2;
 | 
						|
 | 
						|
    switch (offset) {
 | 
						|
    case DPDMA_ISR:
 | 
						|
        s->registers[DPDMA_ISR] &= ~value;
 | 
						|
        xlnx_dpdma_update_irq(s);
 | 
						|
        break;
 | 
						|
    case DPDMA_IEN:
 | 
						|
        s->registers[DPDMA_IMR] &= ~value;
 | 
						|
        break;
 | 
						|
    case DPDMA_IDS:
 | 
						|
        s->registers[DPDMA_IMR] |= value;
 | 
						|
        break;
 | 
						|
    case DPDMA_EISR:
 | 
						|
        s->registers[DPDMA_EISR] &= ~value;
 | 
						|
        xlnx_dpdma_update_irq(s);
 | 
						|
        break;
 | 
						|
    case DPDMA_EIEN:
 | 
						|
        s->registers[DPDMA_EIMR] &= ~value;
 | 
						|
        break;
 | 
						|
    case DPDMA_EIDS:
 | 
						|
        s->registers[DPDMA_EIMR] |= value;
 | 
						|
        break;
 | 
						|
    case DPDMA_IMR:
 | 
						|
    case DPDMA_EIMR:
 | 
						|
    case DPDMA_DSCR_NEXT_ADDRE_CH(0):
 | 
						|
    case DPDMA_DSCR_NEXT_ADDRE_CH(1):
 | 
						|
    case DPDMA_DSCR_NEXT_ADDRE_CH(2):
 | 
						|
    case DPDMA_DSCR_NEXT_ADDRE_CH(3):
 | 
						|
    case DPDMA_DSCR_NEXT_ADDRE_CH(4):
 | 
						|
    case DPDMA_DSCR_NEXT_ADDRE_CH(5):
 | 
						|
    case DPDMA_DSCR_NEXT_ADDR_CH(0):
 | 
						|
    case DPDMA_DSCR_NEXT_ADDR_CH(1):
 | 
						|
    case DPDMA_DSCR_NEXT_ADDR_CH(2):
 | 
						|
    case DPDMA_DSCR_NEXT_ADDR_CH(3):
 | 
						|
    case DPDMA_DSCR_NEXT_ADDR_CH(4):
 | 
						|
    case DPDMA_DSCR_NEXT_ADDR_CH(5):
 | 
						|
    case DPDMA_PYLD_CUR_ADDRE_CH(0):
 | 
						|
    case DPDMA_PYLD_CUR_ADDRE_CH(1):
 | 
						|
    case DPDMA_PYLD_CUR_ADDRE_CH(2):
 | 
						|
    case DPDMA_PYLD_CUR_ADDRE_CH(3):
 | 
						|
    case DPDMA_PYLD_CUR_ADDRE_CH(4):
 | 
						|
    case DPDMA_PYLD_CUR_ADDRE_CH(5):
 | 
						|
    case DPDMA_PYLD_CUR_ADDR_CH(0):
 | 
						|
    case DPDMA_PYLD_CUR_ADDR_CH(1):
 | 
						|
    case DPDMA_PYLD_CUR_ADDR_CH(2):
 | 
						|
    case DPDMA_PYLD_CUR_ADDR_CH(3):
 | 
						|
    case DPDMA_PYLD_CUR_ADDR_CH(4):
 | 
						|
    case DPDMA_PYLD_CUR_ADDR_CH(5):
 | 
						|
    case DPDMA_STATUS_CH(0):
 | 
						|
    case DPDMA_STATUS_CH(1):
 | 
						|
    case DPDMA_STATUS_CH(2):
 | 
						|
    case DPDMA_STATUS_CH(3):
 | 
						|
    case DPDMA_STATUS_CH(4):
 | 
						|
    case DPDMA_STATUS_CH(5):
 | 
						|
    case DPDMA_VDO_CH(0):
 | 
						|
    case DPDMA_VDO_CH(1):
 | 
						|
    case DPDMA_VDO_CH(2):
 | 
						|
    case DPDMA_VDO_CH(3):
 | 
						|
    case DPDMA_VDO_CH(4):
 | 
						|
    case DPDMA_VDO_CH(5):
 | 
						|
    case DPDMA_PYLD_SZ_CH(0):
 | 
						|
    case DPDMA_PYLD_SZ_CH(1):
 | 
						|
    case DPDMA_PYLD_SZ_CH(2):
 | 
						|
    case DPDMA_PYLD_SZ_CH(3):
 | 
						|
    case DPDMA_PYLD_SZ_CH(4):
 | 
						|
    case DPDMA_PYLD_SZ_CH(5):
 | 
						|
    case DPDMA_DSCR_ID_CH(0):
 | 
						|
    case DPDMA_DSCR_ID_CH(1):
 | 
						|
    case DPDMA_DSCR_ID_CH(2):
 | 
						|
    case DPDMA_DSCR_ID_CH(3):
 | 
						|
    case DPDMA_DSCR_ID_CH(4):
 | 
						|
    case DPDMA_DSCR_ID_CH(5):
 | 
						|
        /*
 | 
						|
         * Trying to write to a read only register..
 | 
						|
         */
 | 
						|
        break;
 | 
						|
    case DPDMA_GBL:
 | 
						|
        /*
 | 
						|
         * This is a write only register so it's read as zero in the read
 | 
						|
         * callback.
 | 
						|
         * We store the value anyway so we can know if the channel is
 | 
						|
         * enabled.
 | 
						|
         */
 | 
						|
        s->registers[offset] |= value & 0x00000FFF;
 | 
						|
        break;
 | 
						|
    case DPDMA_DSCR_STRT_ADDRE_CH(0):
 | 
						|
    case DPDMA_DSCR_STRT_ADDRE_CH(1):
 | 
						|
    case DPDMA_DSCR_STRT_ADDRE_CH(2):
 | 
						|
    case DPDMA_DSCR_STRT_ADDRE_CH(3):
 | 
						|
    case DPDMA_DSCR_STRT_ADDRE_CH(4):
 | 
						|
    case DPDMA_DSCR_STRT_ADDRE_CH(5):
 | 
						|
        value &= 0x0000FFFF;
 | 
						|
        s->registers[offset] = value;
 | 
						|
        break;
 | 
						|
    case DPDMA_CNTL_CH(0):
 | 
						|
        s->registers[DPDMA_GBL] &= ~DPDMA_GBL_TRG_CH(0);
 | 
						|
        value &= 0x3FFFFFFF;
 | 
						|
        s->registers[offset] = value;
 | 
						|
        break;
 | 
						|
    case DPDMA_CNTL_CH(1):
 | 
						|
        s->registers[DPDMA_GBL] &= ~DPDMA_GBL_TRG_CH(1);
 | 
						|
        value &= 0x3FFFFFFF;
 | 
						|
        s->registers[offset] = value;
 | 
						|
        break;
 | 
						|
    case DPDMA_CNTL_CH(2):
 | 
						|
        s->registers[DPDMA_GBL] &= ~DPDMA_GBL_TRG_CH(2);
 | 
						|
        value &= 0x3FFFFFFF;
 | 
						|
        s->registers[offset] = value;
 | 
						|
        break;
 | 
						|
    case DPDMA_CNTL_CH(3):
 | 
						|
        s->registers[DPDMA_GBL] &= ~DPDMA_GBL_TRG_CH(3);
 | 
						|
        value &= 0x3FFFFFFF;
 | 
						|
        s->registers[offset] = value;
 | 
						|
        break;
 | 
						|
    case DPDMA_CNTL_CH(4):
 | 
						|
        s->registers[DPDMA_GBL] &= ~DPDMA_GBL_TRG_CH(4);
 | 
						|
        value &= 0x3FFFFFFF;
 | 
						|
        s->registers[offset] = value;
 | 
						|
        break;
 | 
						|
    case DPDMA_CNTL_CH(5):
 | 
						|
        s->registers[DPDMA_GBL] &= ~DPDMA_GBL_TRG_CH(5);
 | 
						|
        value &= 0x3FFFFFFF;
 | 
						|
        s->registers[offset] = value;
 | 
						|
        break;
 | 
						|
    default:
 | 
						|
        assert(offset <= (0xFFC >> 2));
 | 
						|
        s->registers[offset] = value;
 | 
						|
        break;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static const MemoryRegionOps dma_ops = {
 | 
						|
    .read = xlnx_dpdma_read,
 | 
						|
    .write = xlnx_dpdma_write,
 | 
						|
    .endianness = DEVICE_NATIVE_ENDIAN,
 | 
						|
    .valid = {
 | 
						|
        .min_access_size = 4,
 | 
						|
        .max_access_size = 4,
 | 
						|
    },
 | 
						|
    .impl = {
 | 
						|
        .min_access_size = 4,
 | 
						|
        .max_access_size = 4,
 | 
						|
    },
 | 
						|
};
 | 
						|
 | 
						|
static void xlnx_dpdma_init(Object *obj)
 | 
						|
{
 | 
						|
    SysBusDevice *sbd = SYS_BUS_DEVICE(obj);
 | 
						|
    XlnxDPDMAState *s = XLNX_DPDMA(obj);
 | 
						|
 | 
						|
    memory_region_init_io(&s->iomem, obj, &dma_ops, s,
 | 
						|
                          TYPE_XLNX_DPDMA, 0x1000);
 | 
						|
    sysbus_init_mmio(sbd, &s->iomem);
 | 
						|
    sysbus_init_irq(sbd, &s->irq);
 | 
						|
}
 | 
						|
 | 
						|
static void xlnx_dpdma_reset(DeviceState *dev)
 | 
						|
{
 | 
						|
    XlnxDPDMAState *s = XLNX_DPDMA(dev);
 | 
						|
    size_t i;
 | 
						|
 | 
						|
    memset(s->registers, 0, sizeof(s->registers));
 | 
						|
    s->registers[DPDMA_IMR] =  0x07FFFFFF;
 | 
						|
    s->registers[DPDMA_EIMR] = 0xFFFFFFFF;
 | 
						|
    s->registers[DPDMA_ALC0_MIN] = 0x0000FFFF;
 | 
						|
    s->registers[DPDMA_ALC1_MIN] = 0x0000FFFF;
 | 
						|
 | 
						|
    for (i = 0; i < 6; i++) {
 | 
						|
        s->data[i] = NULL;
 | 
						|
        s->operation_finished[i] = true;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static void xlnx_dpdma_class_init(ObjectClass *oc, void *data)
 | 
						|
{
 | 
						|
    DeviceClass *dc = DEVICE_CLASS(oc);
 | 
						|
 | 
						|
    dc->vmsd = &vmstate_xlnx_dpdma;
 | 
						|
    dc->reset = xlnx_dpdma_reset;
 | 
						|
}
 | 
						|
 | 
						|
static const TypeInfo xlnx_dpdma_info = {
 | 
						|
    .name          = TYPE_XLNX_DPDMA,
 | 
						|
    .parent        = TYPE_SYS_BUS_DEVICE,
 | 
						|
    .instance_size = sizeof(XlnxDPDMAState),
 | 
						|
    .instance_init = xlnx_dpdma_init,
 | 
						|
    .class_init    = xlnx_dpdma_class_init,
 | 
						|
};
 | 
						|
 | 
						|
static void xlnx_dpdma_register_types(void)
 | 
						|
{
 | 
						|
    type_register_static(&xlnx_dpdma_info);
 | 
						|
}
 | 
						|
 | 
						|
static MemTxResult xlnx_dpdma_read_descriptor(XlnxDPDMAState *s,
 | 
						|
                                              uint64_t desc_addr,
 | 
						|
                                              DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    MemTxResult res = dma_memory_read(&address_space_memory, desc_addr,
 | 
						|
                                      desc, sizeof(DPDMADescriptor),
 | 
						|
                                      MEMTXATTRS_UNSPECIFIED);
 | 
						|
    if (res) {
 | 
						|
        return res;
 | 
						|
    }
 | 
						|
 | 
						|
    /* Convert from LE into host endianness.  */
 | 
						|
    desc->control = le32_to_cpu(desc->control);
 | 
						|
    desc->descriptor_id = le32_to_cpu(desc->descriptor_id);
 | 
						|
    desc->xfer_size = le32_to_cpu(desc->xfer_size);
 | 
						|
    desc->line_size_stride = le32_to_cpu(desc->line_size_stride);
 | 
						|
    desc->timestamp_lsb = le32_to_cpu(desc->timestamp_lsb);
 | 
						|
    desc->timestamp_msb = le32_to_cpu(desc->timestamp_msb);
 | 
						|
    desc->address_extension = le32_to_cpu(desc->address_extension);
 | 
						|
    desc->next_descriptor = le32_to_cpu(desc->next_descriptor);
 | 
						|
    desc->source_address = le32_to_cpu(desc->source_address);
 | 
						|
    desc->address_extension_23 = le32_to_cpu(desc->address_extension_23);
 | 
						|
    desc->address_extension_45 = le32_to_cpu(desc->address_extension_45);
 | 
						|
    desc->source_address2 = le32_to_cpu(desc->source_address2);
 | 
						|
    desc->source_address3 = le32_to_cpu(desc->source_address3);
 | 
						|
    desc->source_address4 = le32_to_cpu(desc->source_address4);
 | 
						|
    desc->source_address5 = le32_to_cpu(desc->source_address5);
 | 
						|
    desc->crc = le32_to_cpu(desc->crc);
 | 
						|
 | 
						|
    return res;
 | 
						|
}
 | 
						|
 | 
						|
static MemTxResult xlnx_dpdma_write_descriptor(uint64_t desc_addr,
 | 
						|
                                               DPDMADescriptor *desc)
 | 
						|
{
 | 
						|
    DPDMADescriptor tmp_desc = *desc;
 | 
						|
 | 
						|
    /* Convert from host endianness into LE.  */
 | 
						|
    tmp_desc.control = cpu_to_le32(tmp_desc.control);
 | 
						|
    tmp_desc.descriptor_id = cpu_to_le32(tmp_desc.descriptor_id);
 | 
						|
    tmp_desc.xfer_size = cpu_to_le32(tmp_desc.xfer_size);
 | 
						|
    tmp_desc.line_size_stride = cpu_to_le32(tmp_desc.line_size_stride);
 | 
						|
    tmp_desc.timestamp_lsb = cpu_to_le32(tmp_desc.timestamp_lsb);
 | 
						|
    tmp_desc.timestamp_msb = cpu_to_le32(tmp_desc.timestamp_msb);
 | 
						|
    tmp_desc.address_extension = cpu_to_le32(tmp_desc.address_extension);
 | 
						|
    tmp_desc.next_descriptor = cpu_to_le32(tmp_desc.next_descriptor);
 | 
						|
    tmp_desc.source_address = cpu_to_le32(tmp_desc.source_address);
 | 
						|
    tmp_desc.address_extension_23 = cpu_to_le32(tmp_desc.address_extension_23);
 | 
						|
    tmp_desc.address_extension_45 = cpu_to_le32(tmp_desc.address_extension_45);
 | 
						|
    tmp_desc.source_address2 = cpu_to_le32(tmp_desc.source_address2);
 | 
						|
    tmp_desc.source_address3 = cpu_to_le32(tmp_desc.source_address3);
 | 
						|
    tmp_desc.source_address4 = cpu_to_le32(tmp_desc.source_address4);
 | 
						|
    tmp_desc.source_address5 = cpu_to_le32(tmp_desc.source_address5);
 | 
						|
    tmp_desc.crc = cpu_to_le32(tmp_desc.crc);
 | 
						|
 | 
						|
    return dma_memory_write(&address_space_memory, desc_addr, &tmp_desc,
 | 
						|
                            sizeof(DPDMADescriptor), MEMTXATTRS_UNSPECIFIED);
 | 
						|
}
 | 
						|
 | 
						|
size_t xlnx_dpdma_start_operation(XlnxDPDMAState *s, uint8_t channel,
 | 
						|
                                    bool one_desc)
 | 
						|
{
 | 
						|
    uint64_t desc_addr;
 | 
						|
    uint64_t source_addr[6];
 | 
						|
    DPDMADescriptor desc;
 | 
						|
    bool done = false;
 | 
						|
    size_t ptr = 0;
 | 
						|
 | 
						|
    assert(channel <= 5);
 | 
						|
 | 
						|
    DPRINTF("start dpdma channel 0x%" PRIX8 "\n", channel);
 | 
						|
 | 
						|
    if (!xlnx_dpdma_is_channel_triggered(s, channel)) {
 | 
						|
        DPRINTF("Channel isn't triggered..\n");
 | 
						|
        return 0;
 | 
						|
    }
 | 
						|
 | 
						|
    if (!xlnx_dpdma_is_channel_enabled(s, channel)) {
 | 
						|
        DPRINTF("Channel isn't enabled..\n");
 | 
						|
        return 0;
 | 
						|
    }
 | 
						|
 | 
						|
    if (xlnx_dpdma_is_channel_paused(s, channel)) {
 | 
						|
        DPRINTF("Channel is paused..\n");
 | 
						|
        return 0;
 | 
						|
    }
 | 
						|
 | 
						|
    do {
 | 
						|
        if ((s->operation_finished[channel])
 | 
						|
          || xlnx_dpdma_is_channel_retriggered(s, channel)) {
 | 
						|
            desc_addr = xlnx_dpdma_descriptor_start_address(s, channel);
 | 
						|
            s->operation_finished[channel] = false;
 | 
						|
        } else {
 | 
						|
            desc_addr = xlnx_dpdma_descriptor_next_address(s, channel);
 | 
						|
        }
 | 
						|
 | 
						|
        if (xlnx_dpdma_read_descriptor(s, desc_addr, &desc)) {
 | 
						|
            s->registers[DPDMA_EISR] |= ((1 << 1) << channel);
 | 
						|
            xlnx_dpdma_update_irq(s);
 | 
						|
            s->operation_finished[channel] = true;
 | 
						|
            DPRINTF("Can't get the descriptor.\n");
 | 
						|
            break;
 | 
						|
        }
 | 
						|
 | 
						|
        xlnx_dpdma_update_desc_info(s, channel, &desc);
 | 
						|
 | 
						|
#ifdef DEBUG_DPDMA
 | 
						|
        xlnx_dpdma_dump_descriptor(&desc);
 | 
						|
#endif
 | 
						|
 | 
						|
        DPRINTF("location of the descriptor: %" PRIx64 "\n", desc_addr);
 | 
						|
        if (!xlnx_dpdma_desc_is_valid(&desc)) {
 | 
						|
            s->registers[DPDMA_EISR] |= ((1 << 7) << channel);
 | 
						|
            xlnx_dpdma_update_irq(s);
 | 
						|
            s->operation_finished[channel] = true;
 | 
						|
            DPRINTF("Invalid descriptor..\n");
 | 
						|
            break;
 | 
						|
        }
 | 
						|
 | 
						|
        if (xlnx_dpdma_desc_crc_enabled(&desc)
 | 
						|
            && !xlnx_dpdma_desc_check_crc(&desc)) {
 | 
						|
            s->registers[DPDMA_EISR] |= ((1 << 13) << channel);
 | 
						|
            xlnx_dpdma_update_irq(s);
 | 
						|
            s->operation_finished[channel] = true;
 | 
						|
            DPRINTF("Bad CRC for descriptor..\n");
 | 
						|
            break;
 | 
						|
        }
 | 
						|
 | 
						|
        if (xlnx_dpdma_desc_is_already_done(&desc)
 | 
						|
            && !xlnx_dpdma_desc_ignore_done_bit(&desc)) {
 | 
						|
            /* We are trying to process an already processed descriptor. */
 | 
						|
            s->registers[DPDMA_EISR] |= ((1 << 25) << channel);
 | 
						|
            xlnx_dpdma_update_irq(s);
 | 
						|
            s->operation_finished[channel] = true;
 | 
						|
            DPRINTF("Already processed descriptor..\n");
 | 
						|
            break;
 | 
						|
        }
 | 
						|
 | 
						|
        done = xlnx_dpdma_desc_is_last(&desc)
 | 
						|
             || xlnx_dpdma_desc_is_last_of_frame(&desc);
 | 
						|
 | 
						|
        s->operation_finished[channel] = done;
 | 
						|
        if (s->data[channel]) {
 | 
						|
            int64_t transfer_len = xlnx_dpdma_desc_get_transfer_size(&desc);
 | 
						|
            uint32_t line_size = xlnx_dpdma_desc_get_line_size(&desc);
 | 
						|
            uint32_t line_stride = xlnx_dpdma_desc_get_line_stride(&desc);
 | 
						|
            if (xlnx_dpdma_desc_is_contiguous(&desc)) {
 | 
						|
                source_addr[0] = xlnx_dpdma_desc_get_source_address(&desc, 0);
 | 
						|
                while (transfer_len != 0) {
 | 
						|
                    if (dma_memory_read(&address_space_memory,
 | 
						|
                                        source_addr[0],
 | 
						|
                                        &s->data[channel][ptr],
 | 
						|
                                        line_size,
 | 
						|
                                        MEMTXATTRS_UNSPECIFIED)) {
 | 
						|
                        s->registers[DPDMA_ISR] |= ((1 << 12) << channel);
 | 
						|
                        xlnx_dpdma_update_irq(s);
 | 
						|
                        DPRINTF("Can't get data.\n");
 | 
						|
                        break;
 | 
						|
                    }
 | 
						|
                    ptr += line_size;
 | 
						|
                    transfer_len -= line_size;
 | 
						|
                    source_addr[0] += line_stride;
 | 
						|
                }
 | 
						|
            } else {
 | 
						|
                DPRINTF("Source address:\n");
 | 
						|
                int frag;
 | 
						|
                for (frag = 0; frag < 5; frag++) {
 | 
						|
                    source_addr[frag] =
 | 
						|
                          xlnx_dpdma_desc_get_source_address(&desc, frag);
 | 
						|
                    DPRINTF("Fragment %u: %" PRIx64 "\n", frag + 1,
 | 
						|
                            source_addr[frag]);
 | 
						|
                }
 | 
						|
 | 
						|
                frag = 0;
 | 
						|
                while ((transfer_len < 0) && (frag < 5)) {
 | 
						|
                    size_t fragment_len = DPDMA_FRAG_MAX_SZ
 | 
						|
                                    - (source_addr[frag] % DPDMA_FRAG_MAX_SZ);
 | 
						|
 | 
						|
                    if (dma_memory_read(&address_space_memory,
 | 
						|
                                        source_addr[frag],
 | 
						|
                                        &(s->data[channel][ptr]),
 | 
						|
                                        fragment_len,
 | 
						|
                                        MEMTXATTRS_UNSPECIFIED)) {
 | 
						|
                        s->registers[DPDMA_ISR] |= ((1 << 12) << channel);
 | 
						|
                        xlnx_dpdma_update_irq(s);
 | 
						|
                        DPRINTF("Can't get data.\n");
 | 
						|
                        break;
 | 
						|
                    }
 | 
						|
                    ptr += fragment_len;
 | 
						|
                    transfer_len -= fragment_len;
 | 
						|
                    frag += 1;
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        if (xlnx_dpdma_desc_update_enabled(&desc)) {
 | 
						|
            /* The descriptor need to be updated when it's completed. */
 | 
						|
            DPRINTF("update the descriptor with the done flag set.\n");
 | 
						|
            xlnx_dpdma_desc_set_done(&desc);
 | 
						|
            if (xlnx_dpdma_write_descriptor(desc_addr, &desc)) {
 | 
						|
                DPRINTF("Can't write the descriptor.\n");
 | 
						|
                /* TODO: check hardware behaviour for memory write failure */
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        if (xlnx_dpdma_desc_completion_interrupt(&desc)) {
 | 
						|
            DPRINTF("completion interrupt enabled!\n");
 | 
						|
            s->registers[DPDMA_ISR] |= (1 << channel);
 | 
						|
            xlnx_dpdma_update_irq(s);
 | 
						|
        }
 | 
						|
 | 
						|
    } while (!done && !one_desc);
 | 
						|
 | 
						|
    return ptr;
 | 
						|
}
 | 
						|
 | 
						|
void xlnx_dpdma_set_host_data_location(XlnxDPDMAState *s, uint8_t channel,
 | 
						|
                                         void *p)
 | 
						|
{
 | 
						|
    if (!s) {
 | 
						|
        qemu_log_mask(LOG_UNIMP, "DPDMA client not attached to valid DPDMA"
 | 
						|
                      " instance\n");
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    assert(channel <= 5);
 | 
						|
    s->data[channel] = p;
 | 
						|
}
 | 
						|
 | 
						|
void xlnx_dpdma_trigger_vsync_irq(XlnxDPDMAState *s)
 | 
						|
{
 | 
						|
    s->registers[DPDMA_ISR] |= (1 << 27);
 | 
						|
    xlnx_dpdma_update_irq(s);
 | 
						|
}
 | 
						|
 | 
						|
type_init(xlnx_dpdma_register_types)
 |