Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 2538
b: refs/heads/master
c: b8acb80
h: refs/heads/master
v: v3
  • Loading branch information
Max Asbock authored and Linus Torvalds committed Jun 22, 2005
1 parent ea15d2d commit addeddf
Show file tree
Hide file tree
Showing 4 changed files with 41 additions and 15 deletions.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: f5ccc842318efcd7c05dee3203dfdbbafae47bd6
refs/heads/master: b8acb808468a88a188d7c5aba3681c583a5785f9
18 changes: 12 additions & 6 deletions trunk/drivers/misc/ibmasm/event.c
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
*/

#include "ibmasm.h"
#include "lowlevel.h"

/*
* ASM service processor event handling routines.
Expand All @@ -34,7 +35,6 @@
* circular buffer.
*/


static void wake_up_event_readers(struct service_processor *sp)
{
struct event_reader *reader;
Expand Down Expand Up @@ -63,7 +63,7 @@ void ibmasm_receive_event(struct service_processor *sp, void *data, unsigned int
spin_lock_irqsave(&sp->lock, flags);
/* copy the event into the next slot in the circular buffer */
event = &buffer->events[buffer->next_index];
memcpy(event->data, data, data_size);
memcpy_fromio(event->data, data, data_size);
event->data_size = data_size;
event->serial_number = buffer->next_serial_number;

Expand Down Expand Up @@ -93,7 +93,10 @@ int ibmasm_get_next_event(struct service_processor *sp, struct event_reader *rea
unsigned int index;
unsigned long flags;

if (wait_event_interruptible(reader->wait, event_available(buffer, reader)))
reader->cancelled = 0;

if (wait_event_interruptible(reader->wait,
event_available(buffer, reader) || reader->cancelled))
return -ERESTARTSYS;

if (!event_available(buffer, reader))
Expand All @@ -116,6 +119,12 @@ int ibmasm_get_next_event(struct service_processor *sp, struct event_reader *rea
return event->data_size;
}

void ibmasm_cancel_next_event(struct event_reader *reader)
{
reader->cancelled = 1;
wake_up_interruptible(&reader->wait);
}

void ibmasm_event_reader_register(struct service_processor *sp, struct event_reader *reader)
{
unsigned long flags;
Expand All @@ -131,8 +140,6 @@ void ibmasm_event_reader_unregister(struct service_processor *sp, struct event_r
{
unsigned long flags;

wake_up_interruptible(&reader->wait);

spin_lock_irqsave(&sp->lock, flags);
list_del(&reader->node);
spin_unlock_irqrestore(&sp->lock, flags);
Expand Down Expand Up @@ -164,6 +171,5 @@ int ibmasm_event_buffer_init(struct service_processor *sp)

void ibmasm_event_buffer_exit(struct service_processor *sp)
{
wake_up_event_readers(sp);
kfree(sp->event_buffer);
}
2 changes: 2 additions & 0 deletions trunk/drivers/misc/ibmasm/ibmasm.h
Original file line number Diff line number Diff line change
Expand Up @@ -108,6 +108,7 @@ struct event_buffer {
};

struct event_reader {
int cancelled;
unsigned int next_serial_number;
wait_queue_head_t wait;
struct list_head node;
Expand Down Expand Up @@ -185,6 +186,7 @@ extern void ibmasm_receive_event(struct service_processor *sp, void *data, unsi
extern void ibmasm_event_reader_register(struct service_processor *sp, struct event_reader *reader);
extern void ibmasm_event_reader_unregister(struct service_processor *sp, struct event_reader *reader);
extern int ibmasm_get_next_event(struct service_processor *sp, struct event_reader *reader);
extern void ibmasm_cancel_next_event(struct event_reader *reader);

/* heartbeat - from SP to OS */
extern void ibmasm_register_panic_notifier(void);
Expand Down
34 changes: 26 additions & 8 deletions trunk/drivers/misc/ibmasm/ibmasmfs.c
Original file line number Diff line number Diff line change
Expand Up @@ -374,6 +374,7 @@ static int event_file_open(struct inode *inode, struct file *file)
ibmasm_event_reader_register(sp, &event_data->reader);

event_data->sp = sp;
event_data->active = 0;
file->private_data = event_data;
return 0;
}
Expand All @@ -391,7 +392,9 @@ static ssize_t event_file_read(struct file *file, char __user *buf, size_t count
{
struct ibmasmfs_event_data *event_data = file->private_data;
struct event_reader *reader = &event_data->reader;
struct service_processor *sp = event_data->sp;
int ret;
unsigned long flags;

if (*offset < 0)
return -EINVAL;
Expand All @@ -400,17 +403,32 @@ static ssize_t event_file_read(struct file *file, char __user *buf, size_t count
if (*offset != 0)
return 0;

ret = ibmasm_get_next_event(event_data->sp, reader);
spin_lock_irqsave(&sp->lock, flags);
if (event_data->active) {
spin_unlock_irqrestore(&sp->lock, flags);
return -EBUSY;
}
event_data->active = 1;
spin_unlock_irqrestore(&sp->lock, flags);

ret = ibmasm_get_next_event(sp, reader);
if (ret <= 0)
return ret;
goto out;

if (count < reader->data_size)
return -EINVAL;
if (count < reader->data_size) {
ret = -EINVAL;
goto out;
}

if (copy_to_user(buf, reader->data, reader->data_size))
return -EFAULT;
if (copy_to_user(buf, reader->data, reader->data_size)) {
ret = -EFAULT;
goto out;
}
ret = reader->data_size;

return reader->data_size;
out:
event_data->active = 0;
return ret;
}

static ssize_t event_file_write(struct file *file, const char __user *buf, size_t count, loff_t *offset)
Expand All @@ -424,7 +442,7 @@ static ssize_t event_file_write(struct file *file, const char __user *buf, size_
if (*offset != 0)
return 0;

wake_up_interruptible(&event_data->reader.wait);
ibmasm_cancel_next_event(&event_data->reader);
return 0;
}

Expand Down

0 comments on commit addeddf

Please sign in to comment.