aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/scsi/aacraid/commsup.c100
1 files changed, 81 insertions, 19 deletions
diff --git a/drivers/scsi/aacraid/commsup.c b/drivers/scsi/aacraid/commsup.c
index a1d303f03480..3741be2f4bfa 100644
--- a/drivers/scsi/aacraid/commsup.c
+++ b/drivers/scsi/aacraid/commsup.c
@@ -805,7 +805,6 @@ int aac_command_thread(struct aac_dev * dev)
805{ 805{
806 struct hw_fib *hw_fib, *hw_newfib; 806 struct hw_fib *hw_fib, *hw_newfib;
807 struct fib *fib, *newfib; 807 struct fib *fib, *newfib;
808 struct aac_queue_block *queues = dev->queues;
809 struct aac_fib_context *fibctx; 808 struct aac_fib_context *fibctx;
810 unsigned long flags; 809 unsigned long flags;
811 DECLARE_WAITQUEUE(wait, current); 810 DECLARE_WAITQUEUE(wait, current);
@@ -825,21 +824,22 @@ int aac_command_thread(struct aac_dev * dev)
825 * Let the DPC know it has a place to send the AIF's to. 824 * Let the DPC know it has a place to send the AIF's to.
826 */ 825 */
827 dev->aif_thread = 1; 826 dev->aif_thread = 1;
828 add_wait_queue(&queues->queue[HostNormCmdQueue].cmdready, &wait); 827 add_wait_queue(&dev->queues->queue[HostNormCmdQueue].cmdready, &wait);
829 set_current_state(TASK_INTERRUPTIBLE); 828 set_current_state(TASK_INTERRUPTIBLE);
829 dprintk ((KERN_INFO "aac_command_thread start\n"));
830 while(1) 830 while(1)
831 { 831 {
832 spin_lock_irqsave(queues->queue[HostNormCmdQueue].lock, flags); 832 spin_lock_irqsave(dev->queues->queue[HostNormCmdQueue].lock, flags);
833 while(!list_empty(&(queues->queue[HostNormCmdQueue].cmdq))) { 833 while(!list_empty(&(dev->queues->queue[HostNormCmdQueue].cmdq))) {
834 struct list_head *entry; 834 struct list_head *entry;
835 struct aac_aifcmd * aifcmd; 835 struct aac_aifcmd * aifcmd;
836 836
837 set_current_state(TASK_RUNNING); 837 set_current_state(TASK_RUNNING);
838 838
839 entry = queues->queue[HostNormCmdQueue].cmdq.next; 839 entry = dev->queues->queue[HostNormCmdQueue].cmdq.next;
840 list_del(entry); 840 list_del(entry);
841 841
842 spin_unlock_irqrestore(queues->queue[HostNormCmdQueue].lock, flags); 842 spin_unlock_irqrestore(dev->queues->queue[HostNormCmdQueue].lock, flags);
843 fib = list_entry(entry, struct fib, fiblink); 843 fib = list_entry(entry, struct fib, fiblink);
844 /* 844 /*
845 * We will process the FIB here or pass it to a 845 * We will process the FIB here or pass it to a
@@ -869,9 +869,54 @@ int aac_command_thread(struct aac_dev * dev)
869 869
870 u32 time_now, time_last; 870 u32 time_now, time_last;
871 unsigned long flagv; 871 unsigned long flagv;
872 unsigned num;
873 struct hw_fib ** hw_fib_pool, ** hw_fib_p;
874 struct fib ** fib_pool, ** fib_p;
872 875
873 time_now = jiffies/HZ; 876 time_now = jiffies/HZ;
874 877
878 /*
879 * Warning: no sleep allowed while
880 * holding spinlock. We take the estimate
881 * and pre-allocate a set of fibs outside the
882 * lock.
883 */
884 num = le32_to_cpu(dev->init->AdapterFibsSize)
885 / sizeof(struct hw_fib); /* some extra */
886 spin_lock_irqsave(&dev->fib_lock, flagv);
887 entry = dev->fib_list.next;
888 while (entry != &dev->fib_list) {
889 entry = entry->next;
890 ++num;
891 }
892 spin_unlock_irqrestore(&dev->fib_lock, flagv);
893 hw_fib_pool = NULL;
894 fib_pool = NULL;
895 if (num
896 && ((hw_fib_pool = kmalloc(sizeof(struct hw_fib *) * num, GFP_KERNEL)))
897 && ((fib_pool = kmalloc(sizeof(struct fib *) * num, GFP_KERNEL)))) {
898 hw_fib_p = hw_fib_pool;
899 fib_p = fib_pool;
900 while (hw_fib_p < &hw_fib_pool[num]) {
901 if (!(*(hw_fib_p++) = kmalloc(sizeof(struct hw_fib), GFP_KERNEL))) {
902 --hw_fib_p;
903 break;
904 }
905 if (!(*(fib_p++) = kmalloc(sizeof(struct fib), GFP_KERNEL))) {
906 kfree(*(--hw_fib_p));
907 break;
908 }
909 }
910 if ((num = hw_fib_p - hw_fib_pool) == 0) {
911 kfree(fib_pool);
912 fib_pool = NULL;
913 kfree(hw_fib_pool);
914 hw_fib_pool = NULL;
915 }
916 } else if (hw_fib_pool) {
917 kfree(hw_fib_pool);
918 hw_fib_pool = NULL;
919 }
875 spin_lock_irqsave(&dev->fib_lock, flagv); 920 spin_lock_irqsave(&dev->fib_lock, flagv);
876 entry = dev->fib_list.next; 921 entry = dev->fib_list.next;
877 /* 922 /*
@@ -880,6 +925,8 @@ int aac_command_thread(struct aac_dev * dev)
880 * fib, and then set the event to wake up the 925 * fib, and then set the event to wake up the
881 * thread that is waiting for it. 926 * thread that is waiting for it.
882 */ 927 */
928 hw_fib_p = hw_fib_pool;
929 fib_p = fib_pool;
883 while (entry != &dev->fib_list) { 930 while (entry != &dev->fib_list) {
884 /* 931 /*
885 * Extract the fibctx 932 * Extract the fibctx
@@ -912,9 +959,11 @@ int aac_command_thread(struct aac_dev * dev)
912 * Warning: no sleep allowed while 959 * Warning: no sleep allowed while
913 * holding spinlock 960 * holding spinlock
914 */ 961 */
915 hw_newfib = kmalloc(sizeof(struct hw_fib), GFP_ATOMIC); 962 if (hw_fib_p < &hw_fib_pool[num]) {
916 newfib = kmalloc(sizeof(struct fib), GFP_ATOMIC); 963 hw_newfib = *hw_fib_p;
917 if (newfib && hw_newfib) { 964 *(hw_fib_p++) = NULL;
965 newfib = *fib_p;
966 *(fib_p++) = NULL;
918 /* 967 /*
919 * Make the copy of the FIB 968 * Make the copy of the FIB
920 */ 969 */
@@ -929,15 +978,11 @@ int aac_command_thread(struct aac_dev * dev)
929 fibctx->count++; 978 fibctx->count++;
930 /* 979 /*
931 * Set the event to wake up the 980 * Set the event to wake up the
932 * thread that will waiting. 981 * thread that is waiting.
933 */ 982 */
934 up(&fibctx->wait_sem); 983 up(&fibctx->wait_sem);
935 } else { 984 } else {
936 printk(KERN_WARNING "aifd: didn't allocate NewFib.\n"); 985 printk(KERN_WARNING "aifd: didn't allocate NewFib.\n");
937 if(newfib)
938 kfree(newfib);
939 if(hw_newfib)
940 kfree(hw_newfib);
941 } 986 }
942 entry = entry->next; 987 entry = entry->next;
943 } 988 }
@@ -947,21 +992,38 @@ int aac_command_thread(struct aac_dev * dev)
947 *(__le32 *)hw_fib->data = cpu_to_le32(ST_OK); 992 *(__le32 *)hw_fib->data = cpu_to_le32(ST_OK);
948 fib_adapter_complete(fib, sizeof(u32)); 993 fib_adapter_complete(fib, sizeof(u32));
949 spin_unlock_irqrestore(&dev->fib_lock, flagv); 994 spin_unlock_irqrestore(&dev->fib_lock, flagv);
995 /* Free up the remaining resources */
996 hw_fib_p = hw_fib_pool;
997 fib_p = fib_pool;
998 while (hw_fib_p < &hw_fib_pool[num]) {
999 if (*hw_fib_p)
1000 kfree(*hw_fib_p);
1001 if (*fib_p)
1002 kfree(*fib_p);
1003 ++fib_p;
1004 ++hw_fib_p;
1005 }
1006 if (hw_fib_pool)
1007 kfree(hw_fib_pool);
1008 if (fib_pool)
1009 kfree(fib_pool);
950 } 1010 }
951 spin_lock_irqsave(queues->queue[HostNormCmdQueue].lock, flags);
952 kfree(fib); 1011 kfree(fib);
1012 spin_lock_irqsave(dev->queues->queue[HostNormCmdQueue].lock, flags);
953 } 1013 }
954 /* 1014 /*
955 * There are no more AIF's 1015 * There are no more AIF's
956 */ 1016 */
957 spin_unlock_irqrestore(queues->queue[HostNormCmdQueue].lock, flags); 1017 spin_unlock_irqrestore(dev->queues->queue[HostNormCmdQueue].lock, flags);
958 schedule(); 1018 schedule();
959 1019
960 if(signal_pending(current)) 1020 if(signal_pending(current))
961 break; 1021 break;
962 set_current_state(TASK_INTERRUPTIBLE); 1022 set_current_state(TASK_INTERRUPTIBLE);
963 } 1023 }
964 remove_wait_queue(&queues->queue[HostNormCmdQueue].cmdready, &wait); 1024 if (dev->queues)
1025 remove_wait_queue(&dev->queues->queue[HostNormCmdQueue].cmdready, &wait);
965 dev->aif_thread = 0; 1026 dev->aif_thread = 0;
966 complete_and_exit(&dev->aif_completion, 0); 1027 complete_and_exit(&dev->aif_completion, 0);
1028 return 0;
967} 1029}