aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--arch/um/drivers/line.c54
1 files changed, 24 insertions, 30 deletions
diff --git a/arch/um/drivers/line.c b/arch/um/drivers/line.c
index a3c39373adb9..46ceb25a9959 100644
--- a/arch/um/drivers/line.c
+++ b/arch/um/drivers/line.c
@@ -774,55 +774,49 @@ void register_winch_irq(int fd, int tty_fd, int pid, struct tty_struct *tty)
774 printk("register_winch_irq - failed to register IRQ\n"); 774 printk("register_winch_irq - failed to register IRQ\n");
775} 775}
776 776
777static void free_winch(struct winch *winch)
778{
779 list_del(&winch->list);
780
781 if(winch->pid != -1)
782 os_kill_process(winch->pid, 1);
783 if(winch->fd != -1)
784 os_close_file(winch->fd);
785
786 free_irq(WINCH_IRQ, winch);
787 kfree(winch);
788}
789
777static void unregister_winch(struct tty_struct *tty) 790static void unregister_winch(struct tty_struct *tty)
778{ 791{
779 struct list_head *ele; 792 struct list_head *ele;
780 struct winch *winch, *found = NULL; 793 struct winch *winch;
781 794
782 spin_lock(&winch_handler_lock); 795 spin_lock(&winch_handler_lock);
796
783 list_for_each(ele, &winch_handlers){ 797 list_for_each(ele, &winch_handlers){
784 winch = list_entry(ele, struct winch, list); 798 winch = list_entry(ele, struct winch, list);
785 if(winch->tty == tty){ 799 if(winch->tty == tty){
786 found = winch; 800 free_winch(winch);
787 break; 801 break;
788 } 802 }
789 } 803 }
790 if(found == NULL)
791 goto err;
792
793 list_del(&winch->list);
794 spin_unlock(&winch_handler_lock);
795
796 if(winch->pid != -1)
797 os_kill_process(winch->pid, 1);
798
799 free_irq(WINCH_IRQ, winch);
800 kfree(winch);
801
802 return;
803err:
804 spin_unlock(&winch_handler_lock); 804 spin_unlock(&winch_handler_lock);
805} 805}
806 806
807/* XXX: No lock as it's an exitcall... is this valid? Depending on cleanup
808 * order... are we sure that nothing else is done on the list? */
809static void winch_cleanup(void) 807static void winch_cleanup(void)
810{ 808{
811 struct list_head *ele; 809 struct list_head *ele, *next;
812 struct winch *winch; 810 struct winch *winch;
813 811
814 list_for_each(ele, &winch_handlers){ 812 spin_lock(&winch_handler_lock);
813
814 list_for_each_safe(ele, next, &winch_handlers){
815 winch = list_entry(ele, struct winch, list); 815 winch = list_entry(ele, struct winch, list);
816 if(winch->fd != -1){ 816 free_winch(winch);
817 /* Why is this different from the above free_irq(),
818 * which deactivates SIGIO? This searches the FD
819 * somewhere else and removes it from the list... */
820 deactivate_fd(winch->fd, WINCH_IRQ);
821 os_close_file(winch->fd);
822 }
823 if(winch->pid != -1)
824 os_kill_process(winch->pid, 1);
825 } 817 }
818
819 spin_unlock(&winch_handler_lock);
826} 820}
827__uml_exitcall(winch_cleanup); 821__uml_exitcall(winch_cleanup);
828 822