[PATCH RFC 3/9] s390x/css: introduce ccw chain interfaces

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Introduce CcwChain structures and helper functions that can be used
to translate a guest ccw program to a user-space ccw program.
The following limitations apply:
1. Support only prefetch enable mode.
2. Support chain for idal(c64) ccws.
3. Detect chain end only by CC and DC flags.

This work prepares a user-space ccw program according to the rules
below:
1. Alloc a 4K memory buffer to store all of the ccw program information.
   * Lower 2k of the buffer are used to store a maximum of 256 ccws, these
     ccws are copied from 'guest ccw program' and placed one after another.
   * Upper 2k of the buffer are used to store a maximum of 256 corresponding
     cda data sets, each having a length of 8 bytes.
2. For TIC ccw.
   * Locate the TIC target ccw inside the ccw area, and calculate its offset.
   * Store the offset to ccw.cda.
3. For Direct ccw.
   * Find the cda entry with the same index as the ccw.
   * Store the user virtual address of the original ccw.cda to the cda entry.
   * Store the offset of the cda entry to ccw.cda.
4. For IDAL ccw.
   * Find the cda entry with the same index as the ccw.
   * Prepare the user-space idaws. Store the virtual address of the idaws to
     the cda entry.
   * Store the offset of the cda entry to ccw.cda.
5. Append a NOOP to the chain end.
6. Expectations for the user-space ccw program I/O result:
   CPA of SCSW should be set to the offset of the ccw area of the current ccw.

Signed-off-by: Xiao Feng Ren <renxiaof@xxxxxxxxxxxxxxxxxx>
---
 hw/s390x/Makefile.objs   |   1 +
 hw/s390x/s390-ccwchain.c | 441 +++++++++++++++++++++++++++++++++++++++++++++++
 hw/s390x/s390-ccwchain.h |  28 +++
 3 files changed, 470 insertions(+)
 create mode 100644 hw/s390x/s390-ccwchain.c
 create mode 100644 hw/s390x/s390-ccwchain.h

diff --git a/hw/s390x/Makefile.objs b/hw/s390x/Makefile.objs
index 2203617..35b5d27 100644
--- a/hw/s390x/Makefile.objs
+++ b/hw/s390x/Makefile.objs
@@ -11,3 +11,4 @@ obj-y += virtio-ccw.o
 obj-y += s390-pci-bus.o s390-pci-inst.o
 obj-y += s390-skeys.o
 obj-$(CONFIG_KVM) += s390-skeys-kvm.o
+obj-y += s390-ccwchain.o
diff --git a/hw/s390x/s390-ccwchain.c b/hw/s390x/s390-ccwchain.c
new file mode 100644
index 0000000..e62869d
--- /dev/null
+++ b/hw/s390x/s390-ccwchain.c
@@ -0,0 +1,441 @@
+ /*
+  * s390 ccwchain interface
+  *
+  * Copyright 2016 IBM Corp.
+  * Author(s): Dong Jia Shi <bjsdjshi@xxxxxxxxxxxxxxxxxx>
+  *            Xiao Feng Ren <renxiaof@xxxxxxxxxxxxxxxxxx>
+  *
+  * This work is licensed under the terms of the GNU GPL, version
+  * 2 or (at your option) any later version. See the COPYING file
+  * in the top-level directory.
+  */
+
+#include "qemu/osdep.h"
+#include "cpu.h"
+#include "s390-ccwchain.h"
+
+#define IDA_SIZE_LOG 12 /* 12 for 4k */
+#define IDA_BLOCK_SIZE (1L << IDA_SIZE_LOG)
+#define CCWCHAIN_LEN_MAX 256
+#define CCW1_SIZE 3 /* sizeof(CCW1) == (1 << 3) */
+#define CDA_ITEM_SIZE 3 /* sizeof(uint64_t) == (1 << 3) */
+
+#define ccw_is_test(_ccw) (((_ccw)->cmd_code & 0x0F) == 0)
+
+#define ccw_is_noop(_ccw) ((_ccw)->cmd_code == CCW_CMD_NOOP)
+
+#define ccw_is_tic(_ccw) ((_ccw)->cmd_code == CCW_CMD_TIC)
+
+#define ccw_is_idal(_ccw) ((_ccw)->flags & CCW_FLAG_IDA)
+
+#define ccw_is_chain(_ccw) ((_ccw)->flags & (CCW_FLAG_CC | CCW_FLAG_DC))
+
+typedef struct IdaWords {
+    uint32_t nr;
+    uint64_t *ida_word;
+} IdaWords;
+
+typedef struct CcwChainBuffer {
+    CCW1 ccw[CCWCHAIN_LEN_MAX];
+    uint64_t cda[CCWCHAIN_LEN_MAX];
+} CcwChainBuffer;
+
+typedef struct CcwChain {
+    QTAILQ_ENTRY(CcwChain) entry;
+    uint32_t nr;
+    hwaddr gpa;
+    CCW1 *ccw;
+    IdaWords *ida_words;
+} CcwChain;
+
+typedef struct CcwChainList {
+    QTAILQ_HEAD(, CcwChain) list;
+    CcwChainBuffer buf;
+    uint32_t nr; /* Number of the CCWs in the whole list. */
+} CcwChainList;
+
+static int ccwchain_translate(CcwChain *chain, CcwChainList *ccwchain_list);
+
+static inline void *ccwchain_hva_get(hwaddr gpa)
+{
+    hwaddr len = 1;
+    return cpu_physical_memory_map(gpa, &len, 1);
+}
+
+static inline void ccwchain_hva_put(void *hva)
+{
+    cpu_physical_memory_unmap(hva, 1, 1, 1);
+}
+
+static inline void ccwchain_idaword_put(IdaWords *ida_words)
+{
+    uint64_t *ida_word;
+    uint32_t cnt;
+
+    ida_word = ida_words->ida_word;
+    cnt = ida_words->nr;
+    while (cnt--) {
+        ccwchain_hva_put((void *)*ida_word);
+        ida_word++;
+    }
+}
+
+static inline void ccwchain_direct_put(CCW1 *ccw, CcwChainList *ccwchain_list)
+{
+    void *cda_hva;
+
+    cda_hva = (void *)ccwchain_list->buf.cda[ccw->cda >> CDA_ITEM_SIZE];
+    ccwchain_hva_put(cda_hva);
+}
+
+static inline uint32_t ccwchain_ccw_offset(CcwChain *chain,
+                                           CcwChainList *ccwchain_list)
+{
+    return (void *)chain->ccw - (void *)(&ccwchain_list->buf.ccw);
+}
+
+static void ccwchain_free(CcwChain *chain, CcwChainList *ccwchain_list)
+{
+    struct IdaWords *ida_words;
+    CCW1 *ccw;
+    uint32_t idx;
+
+    for (idx = 0; idx < chain->nr; idx++) {
+        ccw = chain->ccw + idx;
+        ida_words = chain->ida_words + idx;
+        if (ccw_is_idal(ccw)) {
+            ccwchain_idaword_put(ida_words);
+        } else if (!ccw_is_test(ccw) && !ccw_is_noop(ccw) &&
+                   !ccw_is_tic(ccw)) {
+            ccwchain_direct_put(ccw, ccwchain_list);
+        }
+        g_free(ida_words->ida_word);
+    }
+
+    QTAILQ_REMOVE(&ccwchain_list->list, chain, entry);
+    g_free(chain);
+}
+
+/*
+ * ccwchain_calc_length - calculate the length of the ccwchain.
+ *
+ * This is the chain length not considering any TICs.
+ * You need to do a new round for each TIC target.
+ *
+ * Returns: the length of the ccwchain.
+ */
+static int ccwchain_calc_length(hwaddr gpa)
+{
+    CCW1 *ccw;
+    int cnt;
+
+    cnt = 0;
+    do {
+        ccw = (CCW1 *)ccwchain_hva_get(gpa);
+        if (!ccw) {
+            return -EFAULT;
+        }
+        cnt++;
+
+        if (!ccw_is_chain(ccw)) {
+            /*
+             * An extra count is needed to reserve a space for
+             * appending an extra NOOP to the chain tail.
+             */
+            cnt++;
+            break;
+        }
+
+        gpa = gpa + sizeof(*ccw);
+        ccwchain_hva_put(ccw);
+    } while (cnt < CCWCHAIN_LEN_MAX + 1);
+
+    if (cnt > CCWCHAIN_LEN_MAX) {
+        cnt = 0;
+    }
+    return cnt;
+}
+
+static CcwChain *ccwchain_alloc(int nr)
+{
+    CcwChain *chain;
+    void *data;
+    size_t size;
+
+    size = sizeof(*chain) + sizeof(*chain->ida_words) * nr;
+    chain = g_malloc0(size);
+    data = (uint8_t *)chain + sizeof(*chain);
+    chain->ida_words = (IdaWords *)data;
+    chain->nr = nr;
+
+    return chain;
+}
+
+static CcwChain *ccwchain_copy_from_guest(hwaddr gpa,
+                                          CcwChainList *ccwchain_list)
+{
+    CcwChain *chain;
+    int nr;
+
+    nr = ccwchain_calc_length(gpa);
+    if (nr <= 0) {
+        return NULL;
+    }
+    if ((ccwchain_list->nr + nr) > CCWCHAIN_LEN_MAX) {
+        return NULL;
+    }
+
+    chain = ccwchain_alloc(nr);
+    if (!chain) {
+        return NULL;
+    }
+
+    chain->gpa = gpa;
+    chain->ccw = (void *)(&ccwchain_list->buf) +
+                 ccwchain_list->nr * sizeof(*chain->ccw);
+    cpu_physical_memory_read(chain->gpa,
+                             chain->ccw,
+                             sizeof(*chain->ccw) * chain->nr);
+    ccwchain_list->nr += chain->nr;
+    QTAILQ_INSERT_TAIL(&ccwchain_list->list, chain, entry);
+
+    return chain;
+}
+
+static bool tic_target_chain_exists(CCW1 *tic, CcwChainList *ccwchain_list)
+{
+    CcwChain *chain;
+    uint32_t ccw_head, ccw_tail;
+
+    QTAILQ_FOREACH(chain, &ccwchain_list->list, entry) {
+        ccw_head = chain->gpa;
+        ccw_tail = ccw_head + sizeof(*chain->ccw) * chain->nr;
+
+        if ((ccw_head <= tic->cda) && (tic->cda <= ccw_tail)) {
+            tic->cda = ccwchain_ccw_offset(chain, ccwchain_list);
+            return true;
+        }
+    }
+    return false;
+}
+
+static int ccwchain_translate_loop_tic(CCW1 *tic,
+                                       CcwChainList *ccwchain_list)
+{
+    CcwChain *chain;
+    int ret;
+
+    /* May transfer to an existing chain. */
+    if (tic_target_chain_exists(tic, ccwchain_list)) {
+        return 0;
+    }
+
+    /* It's a new chain then. */
+    chain = ccwchain_copy_from_guest((uint64_t)tic->cda, ccwchain_list);
+    if (!chain) {
+        return -EFAULT;
+    }
+
+    tic->cda = ccwchain_ccw_offset(chain, ccwchain_list);
+    /* Translate the new ccwchain now. */
+    ret = ccwchain_translate(chain, ccwchain_list);
+    if (ret) {
+        ccwchain_free(chain, ccwchain_list);
+    }
+
+    return ret;
+}
+
+static inline uint32_t idal_nr_words(uint64_t gaddr, uint32_t length)
+{
+    return ((gaddr & (IDA_BLOCK_SIZE - 1)) + length + (IDA_BLOCK_SIZE - 1))
+            >> IDA_SIZE_LOG;
+}
+
+static int copy_idaws_from_guest(CCW1 *ccw, IdaWords *ida_words)
+{
+    uint32_t idaws_len;
+    uint64_t data_addr;
+
+    cpu_physical_memory_read(ccw->cda, &data_addr, 8);
+    ida_words->nr = idal_nr_words(data_addr, ccw->count);
+    idaws_len = ida_words->nr * sizeof(*ida_words->ida_word);
+    ida_words->ida_word = g_malloc0(idaws_len);
+    cpu_physical_memory_read(ccw->cda, ida_words->ida_word, idaws_len);
+
+    return 0;
+}
+
+/*
+ * Translate the guest address in the idaws to user-space virtual address.
+ */
+static int translate_idaws(IdaWords *ida_words)
+{
+    uint64_t *ida_word;
+    uint32_t cnt;
+
+    cnt = ida_words->nr;
+    ida_word = ida_words->ida_word;
+    while (cnt--) {
+        *ida_word = (uint64_t)ccwchain_hva_get(be64_to_cpu(*ida_word));
+        if (!(void *)*ida_word) {
+            return -EFAULT;
+        }
+        ida_word++;
+    }
+
+    return 0;
+}
+
+static void ccwchain_update_cda(CCW1 *ccw,
+                               uint64_t cda_data,
+                               CcwChainList *ccwchain_list)
+{
+    uint32_t index;
+
+    index = ((void *)ccw - (void *)&ccwchain_list->buf.ccw) >> CCW1_SIZE;
+    ccwchain_list->buf.cda[index] = cda_data;
+    ccw->cda = index << CDA_ITEM_SIZE;
+}
+
+static int ccwchain_translate_idal(CCW1 *ccw,
+                                   IdaWords *ida_words,
+                                   CcwChainList *ccwchain_list)
+{
+    int ret;
+
+    ret = copy_idaws_from_guest(ccw, ida_words);
+    if (ret) {
+        return ret;
+    }
+
+    ccwchain_update_cda(ccw, (uint64_t)ida_words->ida_word, ccwchain_list);
+
+    return translate_idaws(ida_words);
+}
+
+static int ccwchain_translate_direct(CCW1 *ccw, CcwChainList *ccwchain_list)
+{
+    void *cda;
+
+    cda = ccwchain_hva_get(ccw->cda);
+    if (!cda) {
+        return -EFAULT;
+    }
+
+    ccwchain_update_cda(ccw, (uint64_t)cda, ccwchain_list);
+
+    return 0;
+}
+
+static int ccwchain_translate(CcwChain *chain, CcwChainList *ccwchain_list)
+{
+    CCW1 *ccw;
+    IdaWords *ida_words;
+    int i, ret;
+
+    for (i = 0; i < chain->nr - 1; i++) {
+        ccw = chain->ccw + i;
+        ccw->count = be16_to_cpu(ccw->count);
+        ccw->cda = be32_to_cpu(ccw->cda);
+        if (ccw_is_test(ccw)) {
+            continue;
+        } else if (ccw_is_noop(ccw)) {
+            continue;
+        } else if (ccw_is_tic(ccw)) {
+            ret = ccwchain_translate_loop_tic(ccw, ccwchain_list);
+        } else if (ccw_is_idal(ccw)) {
+            ida_words = chain->ida_words + i;
+            ret = ccwchain_translate_idal(ccw, ida_words, ccwchain_list);
+        } else {
+            ret = ccwchain_translate_direct(ccw, ccwchain_list);
+        }
+        if (ret) {
+            return ret;
+        }
+    }
+
+    /*
+     * CSS may bypass the last ccw via status-modifier and lead to an
+     * unpredictable behavior. To avoid this, we put a NOOP after the
+     * last ccw.
+     */
+    ccw = chain->ccw + i;
+    ccw->cmd_code = CCW_CMD_NOOP;
+    ccw->flags = 0;
+    ccw->count = 0;
+    ccw->cda = 0;
+
+    return 0;
+}
+
+int ccwchain_translate_to_userspace(TransChainData *chain_data)
+{
+    CcwChain *chain = NULL;
+    CcwChainList *ccwchain_list;
+    int ret;
+
+    ccwchain_list = g_malloc0(sizeof(*ccwchain_list));
+    QTAILQ_INIT(&ccwchain_list->list);
+    chain_data->ccwchain_list = (uint64_t)ccwchain_list;
+    chain_data->ccwchain_buf = (uint64_t)&ccwchain_list->buf;
+
+    chain = ccwchain_copy_from_guest(chain_data->cpa_gpa, ccwchain_list);
+    if (!chain) {
+        ccwchain_list_free(chain_data);
+        return -EFAULT;
+    }
+
+    ret = ccwchain_translate(chain, ccwchain_list);
+    if (ret) {
+        ccwchain_list_free(chain_data);
+        return ret;
+    }
+    chain_data->ccwchain_nr = ccwchain_list->nr;
+
+    return ret;
+}
+
+void ccwchain_update_scsw(SCSW *scsw, TransChainData *chain_data)
+{
+    CcwChainList *ccwchain_list;
+    CcwChain *chain;
+    uint32_t ccw_cnt, ccw_idx;
+    hwaddr cpa;
+
+    cpa = 0;
+    ccw_cnt = 0;
+    ccwchain_list = (void *)chain_data->ccwchain_list;
+    ccw_idx = scsw->cpa >> CCW1_SIZE;
+    QTAILQ_FOREACH(chain, &ccwchain_list->list, entry) {
+        if (ccw_cnt + chain->nr > ccw_idx) {
+            /*
+             * TODO:
+             * When the NOOP introduced by us was hit, we should propagate
+             * this information to the guest.
+             */
+            cpa = chain->gpa + (ccw_idx - ccw_cnt) * sizeof(*chain->ccw);
+            break;
+        }
+
+        ccw_cnt += chain->nr;
+    }
+    scsw->cpa = (uint32_t)cpa;
+}
+
+void ccwchain_list_free(TransChainData *chain_data)
+{
+    CcwChainList *ccwchain_list;
+    CcwChain *chain;
+
+    ccwchain_list = (void *)chain_data->ccwchain_list;
+    if (!ccwchain_list) {
+        return;
+    }
+
+    while ((chain = QTAILQ_FIRST(&ccwchain_list->list)) != NULL) {
+        ccwchain_free(chain, ccwchain_list);
+    }
+
+    g_free(ccwchain_list);
+}
diff --git a/hw/s390x/s390-ccwchain.h b/hw/s390x/s390-ccwchain.h
new file mode 100644
index 0000000..b2352de
--- /dev/null
+++ b/hw/s390x/s390-ccwchain.h
@@ -0,0 +1,28 @@
+/*
+ * ccwchain interfaces
+ *
+ * Copyright IBM Corp. 2016
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License (version 2 only)
+ * as published by the Free Software Foundation.
+ *
+ * Author(s): Dong Jia Shi <bjsdjshi@xxxxxxxxxxxxxxxxxx>
+ *            Xiao Feng Ren <renxiaof@xxxxxxxxxxxxxxxxxx>
+ */
+
+#ifndef S390_CCWCHAIN_H
+#define S390_CCWCHAIN_H
+
+typedef struct TransChainData {
+    hwaddr cpa_gpa;
+    uint64_t ccwchain_list;
+    uint64_t ccwchain_buf;
+    uint32_t ccwchain_nr;
+} TransChainData;
+
+int ccwchain_translate_to_userspace(TransChainData *chain_data);
+void ccwchain_update_scsw(SCSW *scsw, TransChainData *chain_data);
+void ccwchain_list_free(TransChainData *chain_data);
+
+#endif
-- 
2.6.6

--
To unsubscribe from this list: send the line "unsubscribe kvm" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [KVM ARM]     [KVM ia64]     [KVM ppc]     [Virtualization Tools]     [Spice Development]     [Libvirt]     [Libvirt Users]     [Linux USB Devel]     [Linux Audio Users]     [Yosemite Questions]     [Linux Kernel]     [Linux SCSI]     [XFree86]
  Powered by Linux