Commit c5c1e830 authored by Martin Schwidefsky's avatar Martin Schwidefsky Committed by Linus Torvalds

[PATCH] s390: z/VM monitor stream

From: Gerald Schaefer <geraldsc@de.ibm.com>

z/VM monitor stream changes:
 - Add monitor control element to deal with end-of-frame records.
Signed-off-by: default avatarMartin Schwidefsky <schwidefsky@de.ibm.com>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 71dff474
Date : 2004-Nov-04 Date : 2004-Nov-26
Author: Gerald Schaefer (geraldsc@de.ibm.com) Author: Gerald Schaefer (geraldsc@de.ibm.com)
...@@ -72,7 +72,8 @@ the same DCSS. The CP command Q MONITOR (Class E privileged) shows the name ...@@ -72,7 +72,8 @@ the same DCSS. The CP command Q MONITOR (Class E privileged) shows the name
of the monitor DCSS, if already defined, and the users connected to the of the monitor DCSS, if already defined, and the users connected to the
*MONITOR service. *MONITOR service.
Refer to the "z/VM Performance" book (SC24-6109-00) on how to create a monitor Refer to the "z/VM Performance" book (SC24-6109-00) on how to create a monitor
DCSS, you need Class E privileges to define and save a DCSS. DCSS if your z/VM doesn't have one already, you need Class E privileges to
define and save a DCSS.
Example: Example:
-------- --------
...@@ -121,20 +122,41 @@ supported. ...@@ -121,20 +122,41 @@ supported.
Read: Read:
----- -----
Reading from the device provides a set of one or more contiguous monitor Reading from the device provides a 12 Byte monitor control element (MCE),
records, there is no control data (unlike the CMS MONWRITE utility output). followed by a set of one or more contiguous monitor records (similar to the
The monitor record layout can be found here (z/VM 5.1): output of the CMS utility MONWRITE without the 4K control blocks). The MCE
http://www.vm.ibm.com/pubs/mon510/index.html contains information on the type of the following record set (sample/event
data), the monitor domains contained within it and the start and end address
Each set of records is exclusively composed of either event records or sample of the record set in the monitor DCSS. The start and end address can be used
records. The end of such a set of records is indicated by a successful read to determine the size of the record set, the end address is the address of the
with a return value of 0 (0-Byte read). last byte of data. The start address is needed to handle "end-of-frame" records
Any received data must be considered invalid until a complete record set was correctly (domain 1, record 13), i.e. it can be used to determine the record
read successfully, including the closing 0-Byte read. Therefore you should start offset relative to a 4K page (frame) boundary.
See "Appendix A: *MONITOR" in the "z/VM Performance" document for a description
of the monitor control element layout. The layout of the monitor records can
be found here (z/VM 5.1): http://www.vm.ibm.com/pubs/mon510/index.html
The layout of the data stream provided by the monreader device is as follows:
...
<0 byte read>
<first MCE> \
<first set of records> |
... |- data set
<last MCE> |
<last set of records> /
<0 byte read>
...
There may be more than one combination of MCE and corresponding record set
within one data set and the end of each data set is indicated by a successful
read with a return value of 0 (0 byte read).
Any received data must be considered invalid until a complete set was
read successfully, including the closing 0 byte read. Therefore you should
always read the complete set into a buffer before processing the data. always read the complete set into a buffer before processing the data.
The maximum size of a set of records can be as large as the size of the The maximum size of a data set can be as large as the size of the
monitor DCSS, so design the buffer adequately or use dynamic memory allocation monitor DCSS, so design the buffer adequately or use dynamic memory allocation.
The size of the monitor DCSS will be printed into syslog after loading the The size of the monitor DCSS will be printed into syslog after loading the
module. You can also use the (Class E privileged) CP command Q NSS MAP to module. You can also use the (Class E privileged) CP command Q NSS MAP to
list all available segments and information about them. list all available segments and information about them.
...@@ -155,7 +177,7 @@ EOVERFLOW: message limit reached, the data read since the last successful ...@@ -155,7 +177,7 @@ EOVERFLOW: message limit reached, the data read since the last successful
In the last case (EOVERFLOW) there may be missing data, in the first two cases In the last case (EOVERFLOW) there may be missing data, in the first two cases
(EIO, EFAULT) there will be missing data. It's up to the application if it will (EIO, EFAULT) there will be missing data. It's up to the application if it will
continue reading subsequent records or rather exit. continue reading subsequent data or rather exit.
Open: Open:
----- -----
...@@ -163,8 +185,8 @@ Only one user is allowed to open the char device. If it is already in use, the ...@@ -163,8 +185,8 @@ Only one user is allowed to open the char device. If it is already in use, the
open function will fail (return a negative value) and set errno to EBUSY. open function will fail (return a negative value) and set errno to EBUSY.
The open function may also fail if an IUCV connection to the *MONITOR service The open function may also fail if an IUCV connection to the *MONITOR service
cannot be established. In this case errno will be set to EIO and an error cannot be established. In this case errno will be set to EIO and an error
message with an IPUSER SEVER code will be printed into syslog. message with an IPUSER SEVER code will be printed into syslog. The IPUSER SEVER
The IPUSER SEVER codes are described in the "z/VM Performance" book. codes are described in the "z/VM Performance" book, Appendix A.
NOTE: NOTE:
----- -----
......
...@@ -238,6 +238,7 @@ mon_send_reply(struct mon_msg *monmsg, struct mon_private *monpriv) ...@@ -238,6 +238,7 @@ mon_send_reply(struct mon_msg *monmsg, struct mon_private *monpriv)
atomic_dec(&monpriv->msglim_count); atomic_dec(&monpriv->msglim_count);
if (likely(!monmsg->msglim_reached)) { if (likely(!monmsg->msglim_reached)) {
monmsg->pos = 0; monmsg->pos = 0;
monmsg->mca_offset = 0;
monpriv->read_index = (monpriv->read_index + 1) % monpriv->read_index = (monpriv->read_index + 1) %
MON_MSGLIM; MON_MSGLIM;
atomic_dec(&monpriv->read_ready); atomic_dec(&monpriv->read_ready);
...@@ -329,6 +330,7 @@ mon_next_message(struct mon_private *monpriv) ...@@ -329,6 +330,7 @@ mon_next_message(struct mon_private *monpriv)
monmsg->replied_msglim = 0; monmsg->replied_msglim = 0;
monmsg->msglim_reached = 0; monmsg->msglim_reached = 0;
monmsg->pos = 0; monmsg->pos = 0;
monmsg->mca_offset = 0;
P_WARNING("read, message limit reached\n"); P_WARNING("read, message limit reached\n");
monpriv->read_index = (monpriv->read_index + 1) % monpriv->read_index = (monpriv->read_index + 1) %
MON_MSGLIM; MON_MSGLIM;
...@@ -501,6 +503,7 @@ mon_read(struct file *filp, char __user *data, size_t count, loff_t *ppos) ...@@ -501,6 +503,7 @@ mon_read(struct file *filp, char __user *data, size_t count, loff_t *ppos)
struct mon_private *monpriv = filp->private_data; struct mon_private *monpriv = filp->private_data;
struct mon_msg *monmsg; struct mon_msg *monmsg;
int ret; int ret;
u32 mce_start;
monmsg = mon_next_message(monpriv); monmsg = mon_next_message(monpriv);
if (IS_ERR(monmsg)) if (IS_ERR(monmsg))
...@@ -520,13 +523,28 @@ mon_read(struct file *filp, char __user *data, size_t count, loff_t *ppos) ...@@ -520,13 +523,28 @@ mon_read(struct file *filp, char __user *data, size_t count, loff_t *ppos)
} }
if (!monmsg->pos) { if (!monmsg->pos) {
monmsg->pos = mon_rec_start(monmsg); monmsg->pos = mon_mca_start(monmsg) + monmsg->mca_offset;
mon_read_debug(monmsg, monpriv); mon_read_debug(monmsg, monpriv);
} }
if (mon_check_mca(monmsg)) if (mon_check_mca(monmsg))
goto reply; goto reply;
if (mon_rec_end(monmsg) > monmsg->pos) { /* read monitor control element (12 bytes) first */
mce_start = mon_mca_start(monmsg) + monmsg->mca_offset;
if ((monmsg->pos >= mce_start) && (monmsg->pos < mce_start + 12)) {
count = min(count, (size_t) mce_start + 12 - monmsg->pos);
ret = copy_to_user(data, (void *) (unsigned long) monmsg->pos,
count);
if (ret)
return -EFAULT;
monmsg->pos += count;
if (monmsg->pos == mce_start + 12)
monmsg->pos = mon_rec_start(monmsg);
goto out_copy;
}
/* read records */
if (monmsg->pos <= mon_rec_end(monmsg)) {
count = min(count, (size_t) mon_rec_end(monmsg) - monmsg->pos count = min(count, (size_t) mon_rec_end(monmsg) - monmsg->pos
+ 1); + 1);
ret = copy_to_user(data, (void *) (unsigned long) monmsg->pos, ret = copy_to_user(data, (void *) (unsigned long) monmsg->pos,
...@@ -534,14 +552,17 @@ mon_read(struct file *filp, char __user *data, size_t count, loff_t *ppos) ...@@ -534,14 +552,17 @@ mon_read(struct file *filp, char __user *data, size_t count, loff_t *ppos)
if (ret) if (ret)
return -EFAULT; return -EFAULT;
monmsg->pos += count; monmsg->pos += count;
*ppos += count; if (monmsg->pos > mon_rec_end(monmsg))
if (mon_rec_end(monmsg) == monmsg->pos)
mon_next_mca(monmsg); mon_next_mca(monmsg);
return count; goto out_copy;
} }
reply: reply:
ret = mon_send_reply(monmsg, monpriv); ret = mon_send_reply(monmsg, monpriv);
return ret; return ret;
out_copy:
*ppos += count;
return count;
} }
static unsigned int static unsigned int
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment