Commit aad83f9d authored by Stephen Rothwell's avatar Stephen Rothwell Committed by Linus Torvalds

[PATCH] ppc64: iSeries last of the cleanups fo the MF code

This last patch is a bit if a mess because it mainly consists of combining
some single use small functions into their callers and rearranging some other
code.

Some intermediate variables are introduced and some code is restructured to
improve its readablility (and hopefully maintainability).

Overall there are no semantic changes.
Signed-off-by: default avatarStephen Rothwell <sfr@canb.auug.org.au>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 0788b35b
...@@ -55,6 +55,7 @@ ...@@ -55,6 +55,7 @@
#include <asm/iSeries/IoHriMainStore.h> #include <asm/iSeries/IoHriMainStore.h>
#include <asm/iSeries/iSeries_proc.h> #include <asm/iSeries/iSeries_proc.h>
#include <asm/iSeries/mf.h> #include <asm/iSeries/mf.h>
#include <asm/iSeries/HvLpEvent.h>
extern void hvlog(char *fmt, ...); extern void hvlog(char *fmt, ...);
......
...@@ -25,24 +25,21 @@ ...@@ -25,24 +25,21 @@
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/ */
#include <asm/iSeries/mf.h>
#include <linux/types.h> #include <linux/types.h>
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/mm.h>
#include <linux/module.h>
#include <linux/completion.h> #include <linux/completion.h>
#include <asm/iSeries/HvLpConfig.h>
#include <linux/slab.h>
#include <linux/delay.h> #include <linux/delay.h>
#include <asm/nvram.h>
#include <asm/time.h>
#include <asm/iSeries/ItSpCommArea.h>
#include <asm/uaccess.h>
#include <linux/dma-mapping.h> #include <linux/dma-mapping.h>
#include <linux/bcd.h> #include <linux/bcd.h>
#include <asm/time.h>
#include <asm/uaccess.h>
#include <asm/iSeries/vio.h> #include <asm/iSeries/vio.h>
#include <asm/iSeries/mf.h>
#include <asm/iSeries/HvLpConfig.h>
#include <asm/iSeries/ItSpCommArea.h>
/* /*
* This is the structure layout for the Machine Facilites LPAR event * This is the structure layout for the Machine Facilites LPAR event
...@@ -198,8 +195,8 @@ static int signal_event(struct pending_event *ev) ...@@ -198,8 +195,8 @@ static int signal_event(struct pending_event *ev)
hv_rc = HvCallEvent_signalLpEvent( hv_rc = HvCallEvent_signalLpEvent(
&pending_event_head->event.hp_lp_event); &pending_event_head->event.hp_lp_event);
if (hv_rc != HvLpEvent_Rc_Good) { if (hv_rc != HvLpEvent_Rc_Good) {
printk(KERN_ERR "mf.c: HvCallEvent_signalLpEvent() failed with %d\n", printk(KERN_ERR "mf.c: HvCallEvent_signalLpEvent() "
(int)hv_rc); "failed with %d\n", (int)hv_rc);
spin_lock_irqsave(&pending_event_spinlock, flags); spin_lock_irqsave(&pending_event_spinlock, flags);
ev1 = pending_event_head; ev1 = pending_event_head;
...@@ -238,13 +235,14 @@ static struct pending_event *new_pending_event(void) ...@@ -238,13 +235,14 @@ static struct pending_event *new_pending_event(void)
pending_event_avail = pending_event_avail->next; pending_event_avail = pending_event_avail->next;
} }
spin_unlock_irqrestore(&pending_event_spinlock, flags); spin_unlock_irqrestore(&pending_event_spinlock, flags);
if (ev == NULL) if (ev == NULL) {
ev = kmalloc(sizeof(struct pending_event),GFP_ATOMIC); ev = kmalloc(sizeof(struct pending_event), GFP_ATOMIC);
if (ev == NULL) { if (ev == NULL) {
printk(KERN_ERR "mf.c: unable to kmalloc %ld bytes\n", printk(KERN_ERR "mf.c: unable to kmalloc %ld bytes\n",
sizeof(struct pending_event)); sizeof(struct pending_event));
return NULL; return NULL;
} }
}
memset(ev, 0, sizeof(struct pending_event)); memset(ev, 0, sizeof(struct pending_event));
hev = &ev->event.hp_lp_event; hev = &ev->event.hp_lp_event;
hev->xFlags.xValid = 1; hev->xFlags.xValid = 1;
...@@ -254,7 +252,7 @@ static struct pending_event *new_pending_event(void) ...@@ -254,7 +252,7 @@ static struct pending_event *new_pending_event(void)
hev->xType = HvLpEvent_Type_MachineFac; hev->xType = HvLpEvent_Type_MachineFac;
hev->xSourceLp = HvLpConfig_getLpIndex(); hev->xSourceLp = HvLpConfig_getLpIndex();
hev->xTargetLp = primary_lp; hev->xTargetLp = primary_lp;
hev->xSizeMinus1 = sizeof(ev->event)-1; hev->xSizeMinus1 = sizeof(ev->event) - 1;
hev->xRc = HvLpEvent_Rc_Good; hev->xRc = HvLpEvent_Rc_Good;
hev->xSourceInstanceId = HvCallEvent_getSourceLpInstanceId(primary_lp, hev->xSourceInstanceId = HvCallEvent_getSourceLpInstanceId(primary_lp,
HvLpEvent_Type_MachineFac); HvLpEvent_Type_MachineFac);
...@@ -373,8 +371,10 @@ static int shutdown(void) ...@@ -373,8 +371,10 @@ static int shutdown(void)
*/ */
static void handle_int(struct io_mf_lp_event *event) static void handle_int(struct io_mf_lp_event *event)
{ {
int free_it = 0; struct ce_msg_data *ce_msg_data;
struct pending_event *two = NULL; struct ce_msg_data *pce_msg_data;
unsigned long flags;
struct pending_event *pev;
/* ack the interrupt */ /* ack the interrupt */
event->hp_lp_event.xRc = HvLpEvent_Rc_Good; event->hp_lp_event.xRc = HvLpEvent_Rc_Good;
...@@ -383,50 +383,43 @@ static void handle_int(struct io_mf_lp_event *event) ...@@ -383,50 +383,43 @@ static void handle_int(struct io_mf_lp_event *event)
/* process interrupt */ /* process interrupt */
switch (event->hp_lp_event.xSubtype) { switch (event->hp_lp_event.xSubtype) {
case 0: /* CE message */ case 0: /* CE message */
switch (event->data.ce_msg.ce_msg[3]) { ce_msg_data = &event->data.ce_msg;
switch (ce_msg_data->ce_msg[3]) {
case 0x5B: /* power control notification */ case 0x5B: /* power control notification */
if ((event->data.ce_msg.ce_msg[5] & 0x20) != 0) { if ((ce_msg_data->ce_msg[5] & 0x20) != 0) {
printk(KERN_INFO "mf.c: Commencing partition shutdown\n"); printk(KERN_INFO "mf.c: Commencing partition shutdown\n");
if (shutdown() == 0) if (shutdown() == 0)
signal_ce_msg_simple(0xDB, NULL); signal_ce_msg_simple(0xDB, NULL);
} }
break; break;
case 0xC0: /* get time */ case 0xC0: /* get time */
if ((pending_event_head == NULL) || spin_lock_irqsave(&pending_event_spinlock, flags);
(pending_event_head->event.data.ce_msg.ce_msg[3] pev = pending_event_head;
!= 0x40)) if (pev != NULL)
pending_event_head = pending_event_head->next;
spin_unlock_irqrestore(&pending_event_spinlock, flags);
if (pev == NULL)
break; break;
free_it = 1; pce_msg_data = &pev->event.data.ce_msg;
if (pending_event_head->event.data.ce_msg.completion != 0) { if (pce_msg_data->ce_msg[3] != 0x40)
ce_msg_comp_hdlr handler = pending_event_head->event.data.ce_msg.completion->handler; break;
void *token = pending_event_head->event.data.ce_msg.completion->token; if (pce_msg_data->completion != NULL) {
ce_msg_comp_hdlr handler =
pce_msg_data->completion->handler;
void *token = pce_msg_data->completion->token;
if (handler != NULL) if (handler != NULL)
(*handler)(token, &(event->data.ce_msg)); (*handler)(token, ce_msg_data);
}
break;
} }
/* remove from queue */
if (free_it == 1) {
unsigned long flags;
spin_lock_irqsave(&pending_event_spinlock, flags); spin_lock_irqsave(&pending_event_spinlock, flags);
if (pending_event_head != NULL) { free_pending_event(pev);
struct pending_event *oldHead =
pending_event_head;
pending_event_head = pending_event_head->next;
two = pending_event_head;
free_pending_event(oldHead);
}
spin_unlock_irqrestore(&pending_event_spinlock, flags); spin_unlock_irqrestore(&pending_event_spinlock, flags);
}
/* send next waiting event */ /* send next waiting event */
if (two != NULL) if (pending_event_head != NULL)
signal_event(NULL); signal_event(NULL);
break; break;
}
break;
case 1: /* IT sys shutdown */ case 1: /* IT sys shutdown */
printk(KERN_INFO "mf.c: Commencing system shutdown\n"); printk(KERN_INFO "mf.c: Commencing system shutdown\n");
shutdown(); shutdown();
...@@ -442,52 +435,57 @@ static void handle_int(struct io_mf_lp_event *event) ...@@ -442,52 +435,57 @@ static void handle_int(struct io_mf_lp_event *event)
static void handle_ack(struct io_mf_lp_event *event) static void handle_ack(struct io_mf_lp_event *event)
{ {
unsigned long flags; unsigned long flags;
struct pending_event * two = NULL; struct pending_event *two = NULL;
unsigned long free_it = 0; unsigned long free_it = 0;
struct ce_msg_data *ce_msg_data;
struct ce_msg_data *pce_msg_data;
struct vsp_rsp_data *rsp;
/* handle current event */ /* handle current event */
if (pending_event_head != NULL) { if (pending_event_head == NULL) {
printk(KERN_ERR "mf.c: stack empty for receiving ack\n");
return;
}
switch (event->hp_lp_event.xSubtype) { switch (event->hp_lp_event.xSubtype) {
case 0: /* CE msg */ case 0: /* CE msg */
if (event->data.ce_msg.ce_msg[3] == 0x40) { ce_msg_data = &event->data.ce_msg;
if (event->data.ce_msg.ce_msg[2] != 0) { if (ce_msg_data->ce_msg[3] != 0x40) {
free_it = 1; free_it = 1;
if (pending_event_head->event.data.ce_msg.completion break;
!= 0) { }
ce_msg_comp_hdlr handler = pending_event_head->event.data.ce_msg.completion->handler; if (ce_msg_data->ce_msg[2] == 0)
void *token = pending_event_head->event.data.ce_msg.completion->token; break;
free_it = 1;
pce_msg_data = &pending_event_head->event.data.ce_msg;
if (pce_msg_data->completion != NULL) {
ce_msg_comp_hdlr handler =
pce_msg_data->completion->handler;
void *token = pce_msg_data->completion->token;
if (handler != NULL) if (handler != NULL)
(*handler)(token, &(event->data.ce_msg)); (*handler)(token, ce_msg_data);
}
} }
} else
free_it = 1;
break; break;
case 4: /* allocate */ case 4: /* allocate */
case 5: /* deallocate */ case 5: /* deallocate */
if (pending_event_head->hdlr != NULL) { if (pending_event_head->hdlr != NULL)
(*pending_event_head->hdlr)((void *)event->hp_lp_event.xCorrelationToken, event->data.alloc.count); (*pending_event_head->hdlr)((void *)event->hp_lp_event.xCorrelationToken, event->data.alloc.count);
}
free_it = 1; free_it = 1;
break; break;
case 6: case 6:
{
struct vsp_rsp_data *rsp = (struct vsp_rsp_data *)event->data.vsp_cmd.token;
if (rsp != NULL) {
if (rsp->response != NULL)
memcpy(rsp->response, &(event->data.vsp_cmd), sizeof(event->data.vsp_cmd));
complete(&rsp->com);
} else
printk(KERN_ERR "mf.c: no rsp\n");
free_it = 1; free_it = 1;
} rsp = (struct vsp_rsp_data *)event->data.vsp_cmd.token;
if (rsp == NULL) {
printk(KERN_ERR "mf.c: no rsp\n");
break; break;
} }
if (rsp->response != NULL)
memcpy(rsp->response, &event->data.vsp_cmd,
sizeof(event->data.vsp_cmd));
complete(&rsp->com);
break;
} }
else
printk(KERN_ERR "mf.c: stack empty for receiving ack\n");
/* remove from queue */ /* remove from queue */
spin_lock_irqsave(&pending_event_spinlock, flags); spin_lock_irqsave(&pending_event_spinlock, flags);
...@@ -618,7 +616,9 @@ void mf_display_src(u32 word) ...@@ -618,7 +616,9 @@ void mf_display_src(u32 word)
{ {
u8 ce[12]; u8 ce[12];
memcpy(ce, "\x00\x00\x00\x4A\x00\x00\x00\x01\x00\x00\x00\x00", 12); memset(ce, 0, sizeof(ce));
ce[3] = 0x4a;
ce[7] = 0x01;
ce[8] = word >> 24; ce[8] = word >> 24;
ce[9] = word >> 16; ce[9] = word >> 16;
ce[10] = word >> 8; ce[10] = word >> 8;
...@@ -677,232 +677,8 @@ void mf_init(void) ...@@ -677,232 +677,8 @@ void mf_init(void)
signal_ce_msg_simple(0x57, NULL); signal_ce_msg_simple(0x57, NULL);
/* initialization complete */ /* initialization complete */
printk(KERN_NOTICE "mf.c: iSeries Linux LPAR Machine Facilities initialized\n"); printk(KERN_NOTICE "mf.c: iSeries Linux LPAR Machine Facilities "
} "initialized\n");
void mf_setSide(char side)
{
u64 new_side;
struct vsp_cmd_data vsp_cmd;
memset(&vsp_cmd, 0, sizeof(vsp_cmd));
switch (side) {
case 'A': new_side = 0;
break;
case 'B': new_side = 1;
break;
case 'C': new_side = 2;
break;
default: new_side = 3;
break;
}
vsp_cmd.sub_data.ipl_type = new_side;
vsp_cmd.cmd = 10;
(void)signal_vsp_instruction(&vsp_cmd);
}
char mf_getSide(void)
{
char return_value = ' ';
int rc = 0;
struct vsp_cmd_data vsp_cmd;
memset(&vsp_cmd, 0, sizeof(vsp_cmd));
vsp_cmd.cmd = 2;
vsp_cmd.sub_data.ipl_type = 0;
mb();
rc = signal_vsp_instruction(&vsp_cmd);
if (rc != 0)
return return_value;
if (vsp_cmd.result_code == 0) {
switch (vsp_cmd.sub_data.ipl_type) {
case 0: return_value = 'A';
break;
case 1: return_value = 'B';
break;
case 2: return_value = 'C';
break;
default: return_value = 'D';
break;
}
}
return return_value;
}
void mf_getSrcHistory(char *buffer, int size)
{
#if 0
struct IplTypeReturnStuff return_stuff;
struct pending_event *ev = new_pending_event();
int rc = 0;
char *pages[4];
pages[0] = kmalloc(4096, GFP_ATOMIC);
pages[1] = kmalloc(4096, GFP_ATOMIC);
pages[2] = kmalloc(4096, GFP_ATOMIC);
pages[3] = kmalloc(4096, GFP_ATOMIC);
if ((ev == NULL) || (pages[0] == NULL) || (pages[1] == NULL)
|| (pages[2] == NULL) || (pages[3] == NULL))
return -ENOMEM;
return_stuff.xType = 0;
return_stuff.xRc = 0;
return_stuff.xDone = 0;
ev->event.hp_lp_event.xSubtype = 6;
ev->event.hp_lp_event.x.xSubtypeData =
subtype_data('M', 'F', 'V', 'I');
ev->event.data.vsp_cmd.xEvent = &return_stuff;
ev->event.data.vsp_cmd.cmd = 4;
ev->event.data.vsp_cmd.lp_index = HvLpConfig_getLpIndex();
ev->event.data.vsp_cmd.result_code = 0xFF;
ev->event.data.vsp_cmd.reserved = 0;
ev->event.data.vsp_cmd.sub_data.page[0] = ISERIES_HV_ADDR(pages[0]);
ev->event.data.vsp_cmd.sub_data.page[1] = ISERIES_HV_ADDR(pages[1]);
ev->event.data.vsp_cmd.sub_data.page[2] = ISERIES_HV_ADDR(pages[2]);
ev->event.data.vsp_cmd.sub_data.page[3] = ISERIES_HV_ADDR(pages[3]);
mb();
if (signal_event(ev) != 0)
return;
while (return_stuff.xDone != 1)
udelay(10);
if (return_stuff.xRc == 0)
memcpy(buffer, pages[0], size);
kfree(pages[0]);
kfree(pages[1]);
kfree(pages[2]);
kfree(pages[3]);
#endif
}
void mf_setCmdLine(const char *cmdline, int size, u64 side)
{
struct vsp_cmd_data vsp_cmd;
dma_addr_t dma_addr = 0;
char *page = dma_alloc_coherent(iSeries_vio_dev, size, &dma_addr,
GFP_ATOMIC);
if (page == NULL) {
printk(KERN_ERR "mf.c: couldn't allocate memory to set command line\n");
return;
}
copy_from_user(page, cmdline, size);
memset(&vsp_cmd, 0, sizeof(vsp_cmd));
vsp_cmd.cmd = 31;
vsp_cmd.sub_data.kern.token = dma_addr;
vsp_cmd.sub_data.kern.address_type = HvLpDma_AddressType_TceIndex;
vsp_cmd.sub_data.kern.side = side;
vsp_cmd.sub_data.kern.length = size;
mb();
(void)signal_vsp_instruction(&vsp_cmd);
dma_free_coherent(iSeries_vio_dev, size, page, dma_addr);
}
int mf_getCmdLine(char *cmdline, int *size, u64 side)
{
struct vsp_cmd_data vsp_cmd;
int rc;
int len = *size;
dma_addr_t dma_addr;
dma_addr = dma_map_single(iSeries_vio_dev, cmdline, len,
DMA_FROM_DEVICE);
memset(cmdline, 0, len);
memset(&vsp_cmd, 0, sizeof(vsp_cmd));
vsp_cmd.cmd = 33;
vsp_cmd.sub_data.kern.token = dma_addr;
vsp_cmd.sub_data.kern.address_type = HvLpDma_AddressType_TceIndex;
vsp_cmd.sub_data.kern.side = side;
vsp_cmd.sub_data.kern.length = len;
mb();
rc = signal_vsp_instruction(&vsp_cmd);
if (rc == 0) {
if (vsp_cmd.result_code == 0)
len = vsp_cmd.sub_data.length_out;
#if 0
else
memcpy(cmdline, "Bad cmdline", 11);
#endif
}
dma_unmap_single(iSeries_vio_dev, dma_addr, *size, DMA_FROM_DEVICE);
return len;
}
int mf_setVmlinuxChunk(const char *buffer, int size, int offset, u64 side)
{
struct vsp_cmd_data vsp_cmd;
int rc;
dma_addr_t dma_addr = 0;
char *page = dma_alloc_coherent(iSeries_vio_dev, size, &dma_addr,
GFP_ATOMIC);
if (page == NULL) {
printk(KERN_ERR "mf.c: couldn't allocate memory to set vmlinux chunk\n");
return -ENOMEM;
}
copy_from_user(page, buffer, size);
memset(&vsp_cmd, 0, sizeof(vsp_cmd));
vsp_cmd.cmd = 30;
vsp_cmd.sub_data.kern.token = dma_addr;
vsp_cmd.sub_data.kern.address_type = HvLpDma_AddressType_TceIndex;
vsp_cmd.sub_data.kern.side = side;
vsp_cmd.sub_data.kern.offset = offset;
vsp_cmd.sub_data.kern.length = size;
mb();
rc = signal_vsp_instruction(&vsp_cmd);
if (rc == 0) {
if (vsp_cmd.result_code == 0)
rc = 0;
else
rc = -ENOMEM;
}
dma_free_coherent(iSeries_vio_dev, size, page, dma_addr);
return rc;
}
int mf_getVmlinuxChunk(char *buffer, int *size, int offset, u64 side)
{
struct vsp_cmd_data vsp_cmd;
int rc;
int len = *size;
dma_addr_t dma_addr;
dma_addr = dma_map_single(iSeries_vio_dev, buffer, len,
DMA_FROM_DEVICE);
memset(buffer, 0, len);
memset(&vsp_cmd, 0, sizeof(vsp_cmd));
vsp_cmd.cmd = 32;
vsp_cmd.sub_data.kern.token = dma_addr;
vsp_cmd.sub_data.kern.address_type = HvLpDma_AddressType_TceIndex;
vsp_cmd.sub_data.kern.side = side;
vsp_cmd.sub_data.kern.offset = offset;
vsp_cmd.sub_data.kern.length = len;
mb();
rc = signal_vsp_instruction(&vsp_cmd);
if (rc == 0) {
if (vsp_cmd.result_code == 0)
*size = vsp_cmd.sub_data.length_out;
else
rc = -ENOMEM;
}
dma_unmap_single(iSeries_vio_dev, dma_addr, len, DMA_FROM_DEVICE);
return rc;
} }
struct rtc_time_data { struct rtc_time_data {
...@@ -932,10 +708,22 @@ int mf_get_rtc(struct rtc_time *tm) ...@@ -932,10 +708,22 @@ int mf_get_rtc(struct rtc_time *tm)
ce_complete.handler = &get_rtc_time_complete; ce_complete.handler = &get_rtc_time_complete;
ce_complete.token = &rtc_data; ce_complete.token = &rtc_data;
rc = signal_ce_msg_simple(0x40, &ce_complete); rc = signal_ce_msg_simple(0x40, &ce_complete);
if (rc == 0) { if (rc)
return rc;
wait_for_completion(&rtc_data.com); wait_for_completion(&rtc_data.com);
tm->tm_wday = 0;
tm->tm_yday = 0;
tm->tm_isdst = 0;
if (rtc_data.rc) {
tm->tm_sec = 0;
tm->tm_min = 0;
tm->tm_hour = 0;
tm->tm_mday = 15;
tm->tm_mon = 5;
tm->tm_year = 52;
return rtc_data.rc;
}
if (rtc_data.rc == 0) {
if ((rtc_data.ce_msg.ce_msg[2] == 0xa9) || if ((rtc_data.ce_msg.ce_msg[2] == 0xa9) ||
(rtc_data.ce_msg.ce_msg[2] == 0xaf)) { (rtc_data.ce_msg.ce_msg[2] == 0xaf)) {
/* TOD clock is not set */ /* TOD clock is not set */
...@@ -973,27 +761,13 @@ int mf_get_rtc(struct rtc_time *tm) ...@@ -973,27 +761,13 @@ int mf_get_rtc(struct rtc_time *tm)
tm->tm_mon = mon; tm->tm_mon = mon;
tm->tm_year = year; tm->tm_year = year;
} }
} else {
rc = rtc_data.rc;
tm->tm_sec = 0;
tm->tm_min = 0;
tm->tm_hour = 0;
tm->tm_mday = 15;
tm->tm_mon = 5;
tm->tm_year = 52;
} return 0;
tm->tm_wday = 0;
tm->tm_yday = 0;
tm->tm_isdst = 0;
}
return rc;
} }
int mf_set_rtc(struct rtc_time *tm) int mf_set_rtc(struct rtc_time *tm)
{ {
char ce_time[12] = "\x00\x00\x00\x41\x00\x00\x00\x00\x00\x00\x00\x00"; char ce_time[12];
u8 day, mon, hour, min, sec, y1, y2; u8 day, mon, hour, min, sec, y1, y2;
unsigned year; unsigned year;
...@@ -1015,6 +789,8 @@ int mf_set_rtc(struct rtc_time *tm) ...@@ -1015,6 +789,8 @@ int mf_set_rtc(struct rtc_time *tm)
BIN_TO_BCD(y1); BIN_TO_BCD(y1);
BIN_TO_BCD(y2); BIN_TO_BCD(y2);
memset(ce_time, 0, sizeof(ce_time));
ce_time[3] = 0x41;
ce_time[4] = y1; ce_time[4] = y1;
ce_time[5] = y2; ce_time[5] = y2;
ce_time[6] = sec; ce_time[6] = sec;
...@@ -1026,34 +802,96 @@ int mf_set_rtc(struct rtc_time *tm) ...@@ -1026,34 +802,96 @@ int mf_set_rtc(struct rtc_time *tm)
return signal_ce_msg(ce_time, NULL); return signal_ce_msg(ce_time, NULL);
} }
#ifdef CONFIG_PROC_FS
static int proc_mf_dump_cmdline(char *page, char **start, off_t off, static int proc_mf_dump_cmdline(char *page, char **start, off_t off,
int count, int *eof, void *data) int count, int *eof, void *data)
{ {
int len = count; int len;
char *p; char *p;
struct vsp_cmd_data vsp_cmd;
int rc;
dma_addr_t dma_addr;
if (off) { /* The HV appears to return no more than 256 bytes of command line */
*eof = 1; if (off >= 256)
return 0; return 0;
} if ((off + count) > 256)
count = 256 - off;
len = mf_getCmdLine(page, &len, (u64)data);
dma_addr = dma_map_single(iSeries_vio_dev, page, off + count,
DMA_FROM_DEVICE);
if (dma_mapping_error(dma_addr))
return -ENOMEM;
memset(page, 0, off + count);
memset(&vsp_cmd, 0, sizeof(vsp_cmd));
vsp_cmd.cmd = 33;
vsp_cmd.sub_data.kern.token = dma_addr;
vsp_cmd.sub_data.kern.address_type = HvLpDma_AddressType_TceIndex;
vsp_cmd.sub_data.kern.side = (u64)data;
vsp_cmd.sub_data.kern.length = off + count;
mb();
rc = signal_vsp_instruction(&vsp_cmd);
dma_unmap_single(iSeries_vio_dev, dma_addr, off + count,
DMA_FROM_DEVICE);
if (rc)
return rc;
if (vsp_cmd.result_code != 0)
return -ENOMEM;
p = page; p = page;
while (len < (count - 1)) { len = 0;
if (!*p || *p == '\n') while (len < (off + count)) {
break; if ((*p == '\0') || (*p == '\n')) {
if (*p == '\0')
*p = '\n';
p++; p++;
len++; len++;
*eof = 1;
break;
} }
*p = '\n';
p++; p++;
*p = 0; len++;
}
return p - page; if (len < off) {
*eof = 1;
len = 0;
}
return len;
} }
#if 0 #if 0
static int mf_getVmlinuxChunk(char *buffer, int *size, int offset, u64 side)
{
struct vsp_cmd_data vsp_cmd;
int rc;
int len = *size;
dma_addr_t dma_addr;
dma_addr = dma_map_single(iSeries_vio_dev, buffer, len,
DMA_FROM_DEVICE);
memset(buffer, 0, len);
memset(&vsp_cmd, 0, sizeof(vsp_cmd));
vsp_cmd.cmd = 32;
vsp_cmd.sub_data.kern.token = dma_addr;
vsp_cmd.sub_data.kern.address_type = HvLpDma_AddressType_TceIndex;
vsp_cmd.sub_data.kern.side = side;
vsp_cmd.sub_data.kern.offset = offset;
vsp_cmd.sub_data.kern.length = len;
mb();
rc = signal_vsp_instruction(&vsp_cmd);
if (rc == 0) {
if (vsp_cmd.result_code == 0)
*size = vsp_cmd.sub_data.length_out;
else
rc = -ENOMEM;
}
dma_unmap_single(iSeries_vio_dev, dma_addr, len, DMA_FROM_DEVICE);
return rc;
}
static int proc_mf_dump_vmlinux(char *page, char **start, off_t off, static int proc_mf_dump_vmlinux(char *page, char **start, off_t off,
int count, int *eof, void *data) int count, int *eof, void *data)
{ {
...@@ -1079,7 +917,28 @@ static int proc_mf_dump_side(char *page, char **start, off_t off, ...@@ -1079,7 +917,28 @@ static int proc_mf_dump_side(char *page, char **start, off_t off,
int count, int *eof, void *data) int count, int *eof, void *data)
{ {
int len; int len;
char mf_current_side = mf_getSide(); char mf_current_side = ' ';
struct vsp_cmd_data vsp_cmd;
memset(&vsp_cmd, 0, sizeof(vsp_cmd));
vsp_cmd.cmd = 2;
vsp_cmd.sub_data.ipl_type = 0;
mb();
if (signal_vsp_instruction(&vsp_cmd) == 0) {
if (vsp_cmd.result_code == 0) {
switch (vsp_cmd.sub_data.ipl_type) {
case 0: mf_current_side = 'A';
break;
case 1: mf_current_side = 'B';
break;
case 2: mf_current_side = 'C';
break;
default: mf_current_side = 'D';
break;
}
}
}
len = sprintf(page, "%c\n", mf_current_side); len = sprintf(page, "%c\n", mf_current_side);
...@@ -1097,30 +956,92 @@ static int proc_mf_dump_side(char *page, char **start, off_t off, ...@@ -1097,30 +956,92 @@ static int proc_mf_dump_side(char *page, char **start, off_t off,
static int proc_mf_change_side(struct file *file, const char __user *buffer, static int proc_mf_change_side(struct file *file, const char __user *buffer,
unsigned long count, void *data) unsigned long count, void *data)
{ {
char stkbuf[10]; char side;
u64 newSide;
struct vsp_cmd_data vsp_cmd;
if (!capable(CAP_SYS_ADMIN)) if (!capable(CAP_SYS_ADMIN))
return -EACCES; return -EACCES;
if (count > (sizeof(stkbuf) - 1)) if (count == 0)
count = sizeof(stkbuf) - 1; return 0;
if (copy_from_user(stkbuf, buffer, count))
if (get_user(side, buffer))
return -EFAULT; return -EFAULT;
stkbuf[count] = 0;
if ((*stkbuf != 'A') && (*stkbuf != 'B') && switch (side) {
(*stkbuf != 'C') && (*stkbuf != 'D')) { case 'A': newSide = 0;
break;
case 'B': newSide = 1;
break;
case 'C': newSide = 2;
break;
case 'D': newSide = 3;
break;
default:
printk(KERN_ERR "mf_proc.c: proc_mf_change_side: invalid side\n"); printk(KERN_ERR "mf_proc.c: proc_mf_change_side: invalid side\n");
return -EINVAL; return -EINVAL;
} }
mf_setSide(*stkbuf); memset(&vsp_cmd, 0, sizeof(vsp_cmd));
vsp_cmd.sub_data.ipl_type = newSide;
vsp_cmd.cmd = 10;
(void)signal_vsp_instruction(&vsp_cmd);
return count; return count;
} }
#if 0
static void mf_getSrcHistory(char *buffer, int size)
{
struct IplTypeReturnStuff return_stuff;
struct pending_event *ev = new_pending_event();
int rc = 0;
char *pages[4];
pages[0] = kmalloc(4096, GFP_ATOMIC);
pages[1] = kmalloc(4096, GFP_ATOMIC);
pages[2] = kmalloc(4096, GFP_ATOMIC);
pages[3] = kmalloc(4096, GFP_ATOMIC);
if ((ev == NULL) || (pages[0] == NULL) || (pages[1] == NULL)
|| (pages[2] == NULL) || (pages[3] == NULL))
return -ENOMEM;
return_stuff.xType = 0;
return_stuff.xRc = 0;
return_stuff.xDone = 0;
ev->event.hp_lp_event.xSubtype = 6;
ev->event.hp_lp_event.x.xSubtypeData =
subtype_data('M', 'F', 'V', 'I');
ev->event.data.vsp_cmd.xEvent = &return_stuff;
ev->event.data.vsp_cmd.cmd = 4;
ev->event.data.vsp_cmd.lp_index = HvLpConfig_getLpIndex();
ev->event.data.vsp_cmd.result_code = 0xFF;
ev->event.data.vsp_cmd.reserved = 0;
ev->event.data.vsp_cmd.sub_data.page[0] = ISERIES_HV_ADDR(pages[0]);
ev->event.data.vsp_cmd.sub_data.page[1] = ISERIES_HV_ADDR(pages[1]);
ev->event.data.vsp_cmd.sub_data.page[2] = ISERIES_HV_ADDR(pages[2]);
ev->event.data.vsp_cmd.sub_data.page[3] = ISERIES_HV_ADDR(pages[3]);
mb();
if (signal_event(ev) != 0)
return;
while (return_stuff.xDone != 1)
udelay(10);
if (return_stuff.xRc == 0)
memcpy(buffer, pages[0], size);
kfree(pages[0]);
kfree(pages[1]);
kfree(pages[2]);
kfree(pages[3]);
}
#endif
static int proc_mf_dump_src(char *page, char **start, off_t off, static int proc_mf_dump_src(char *page, char **start, off_t off,
int count, int *eof, void *data) int count, int *eof, void *data)
{ {
#if 0
int len; int len;
mf_getSrcHistory(page, count); mf_getSrcHistory(page, count);
...@@ -1134,6 +1055,9 @@ static int proc_mf_dump_src(char *page, char **start, off_t off, ...@@ -1134,6 +1055,9 @@ static int proc_mf_dump_src(char *page, char **start, off_t off,
len = count; len = count;
*start = page + off; *start = page + off;
return len; return len;
#else
return 0;
#endif
} }
static int proc_mf_change_src(struct file *file, const char __user *buffer, static int proc_mf_change_src(struct file *file, const char __user *buffer,
...@@ -1162,34 +1086,91 @@ static int proc_mf_change_src(struct file *file, const char __user *buffer, ...@@ -1162,34 +1086,91 @@ static int proc_mf_change_src(struct file *file, const char __user *buffer,
return count; return count;
} }
static int proc_mf_change_cmdline(struct file *file, const char *buffer, static int proc_mf_change_cmdline(struct file *file, const char __user *buffer,
unsigned long count, void *data) unsigned long count, void *data)
{ {
struct vsp_cmd_data vsp_cmd;
dma_addr_t dma_addr;
char *page;
int ret = -EACCES;
if (!capable(CAP_SYS_ADMIN)) if (!capable(CAP_SYS_ADMIN))
return -EACCES; goto out;
mf_setCmdLine(buffer, count, (u64)data); dma_addr = 0;
page = dma_alloc_coherent(iSeries_vio_dev, count, &dma_addr,
GFP_ATOMIC);
ret = -ENOMEM;
if (page == NULL)
goto out;
return count; ret = -EFAULT;
if (copy_from_user(page, buffer, count))
goto out_free;
memset(&vsp_cmd, 0, sizeof(vsp_cmd));
vsp_cmd.cmd = 31;
vsp_cmd.sub_data.kern.token = dma_addr;
vsp_cmd.sub_data.kern.address_type = HvLpDma_AddressType_TceIndex;
vsp_cmd.sub_data.kern.side = (u64)data;
vsp_cmd.sub_data.kern.length = count;
mb();
(void)signal_vsp_instruction(&vsp_cmd);
ret = count;
out_free:
dma_free_coherent(iSeries_vio_dev, count, page, dma_addr);
out:
return ret;
} }
static ssize_t proc_mf_change_vmlinux(struct file *file, static ssize_t proc_mf_change_vmlinux(struct file *file,
const char __user *buf, const char __user *buf,
size_t count, loff_t *ppos) size_t count, loff_t *ppos)
{ {
struct inode * inode = file->f_dentry->d_inode; struct proc_dir_entry *dp = PDE(file->f_dentry->d_inode);
struct proc_dir_entry * dp = PDE(inode); ssize_t rc;
int rc; dma_addr_t dma_addr;
char *page;
struct vsp_cmd_data vsp_cmd;
rc = -EACCES;
if (!capable(CAP_SYS_ADMIN)) if (!capable(CAP_SYS_ADMIN))
return -EACCES; goto out;
rc = mf_setVmlinuxChunk(buf, count, *ppos, (u64)dp->data); dma_addr = 0;
if (rc < 0) page = dma_alloc_coherent(iSeries_vio_dev, count, &dma_addr,
return rc; GFP_ATOMIC);
rc = -ENOMEM;
if (page == NULL) {
printk(KERN_ERR "mf.c: couldn't allocate memory to set vmlinux chunk\n");
goto out;
}
rc = -EFAULT;
if (copy_from_user(page, buf, count))
goto out_free;
*ppos += count; memset(&vsp_cmd, 0, sizeof(vsp_cmd));
vsp_cmd.cmd = 30;
vsp_cmd.sub_data.kern.token = dma_addr;
vsp_cmd.sub_data.kern.address_type = HvLpDma_AddressType_TceIndex;
vsp_cmd.sub_data.kern.side = (u64)dp->data;
vsp_cmd.sub_data.kern.offset = *ppos;
vsp_cmd.sub_data.kern.length = count;
mb();
rc = signal_vsp_instruction(&vsp_cmd);
if (rc)
goto out_free;
rc = -ENOMEM;
if (vsp_cmd.result_code != 0)
goto out_free;
return count; *ppos += count;
rc = count;
out_free:
dma_free_coherent(iSeries_vio_dev, count, page, dma_addr);
out:
return rc;
} }
static struct file_operations proc_vmlinux_operations = { static struct file_operations proc_vmlinux_operations = {
...@@ -1254,3 +1235,5 @@ static int __init mf_proc_init(void) ...@@ -1254,3 +1235,5 @@ static int __init mf_proc_init(void)
} }
__initcall(mf_proc_init); __initcall(mf_proc_init);
#endif /* CONFIG_PROC_FS */
...@@ -24,13 +24,13 @@ ...@@ -24,13 +24,13 @@
* along with this program; if not, write to the Free Software * along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/ */
#ifndef MF_H_INCLUDED #ifndef _ASM_PPC64_ISERIES_MF_H
#define MF_H_INCLUDED #define _ASM_PPC64_ISERIES_MF_H
#include <linux/proc_fs.h> #include <linux/types.h>
#include <asm/iSeries/HvTypes.h> #include <asm/iSeries/HvTypes.h>
#include <asm/iSeries/HvLpEvent.h> #include <asm/iSeries/HvCallEvent.h>
struct rtc_time; struct rtc_time;
...@@ -51,19 +51,7 @@ extern void mf_clear_src(void); ...@@ -51,19 +51,7 @@ extern void mf_clear_src(void);
extern void mf_init(void); extern void mf_init(void);
extern void mf_setSide(char side);
extern char mf_getSide(void);
extern void mf_setCmdLine(const char *cmdline, int size, u64 side);
extern int mf_getCmdLine(char *cmdline, int *size, u64 side);
extern void mf_getSrcHistory(char *buffer, int size);
extern int mf_setVmlinuxChunk(const char *buffer, int size, int offset,
u64 side);
extern int mf_getVmlinuxChunk(char *buffer, int *size, int offset, u64 side);
extern int mf_get_rtc(struct rtc_time *tm); extern int mf_get_rtc(struct rtc_time *tm);
extern int mf_set_rtc(struct rtc_time *tm); extern int mf_set_rtc(struct rtc_time *tm);
#endif /* MF_H_INCLUDED */ #endif /* _ASM_PPC64_ISERIES_MF_H */
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