]> err.no Git - linux-2.6/blobdiff - arch/x86/kernel/tlb_uv.c
Merge branch 'x86-fixes-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git...
[linux-2.6] / arch / x86 / kernel / tlb_uv.c
index f7bc6a6fbe494afce67aab61707d5172ddeddd74..8b8c0d6640fa9ba9ca7da73fac2cb4ee52e7be26 100644 (file)
 #include <linux/kernel.h>
 
 #include <asm/mmu_context.h>
-#include <asm/idle.h>
-#include <asm/genapic.h>
-#include <asm/uv/uv_hub.h>
 #include <asm/uv/uv_mmrs.h>
+#include <asm/uv/uv_hub.h>
 #include <asm/uv/uv_bau.h>
+#include <asm/genapic.h>
+#include <asm/idle.h>
 #include <asm/tsc.h>
+#include <asm/irq_vectors.h>
 
 #include <mach_apic.h>
 
-static struct bau_control **uv_bau_table_bases __read_mostly;
-static int uv_bau_retry_limit __read_mostly;
-static int uv_nshift __read_mostly; /* position of pnode (which is nasid>>1) */
-static unsigned long uv_mmask __read_mostly;
+static struct bau_control      **uv_bau_table_bases __read_mostly;
+static int                     uv_bau_retry_limit __read_mostly;
 
-char *status_table[] = {
-       "IDLE",
-       "ACTIVE",
-       "DESTINATION TIMEOUT",
-       "SOURCE TIMEOUT"
-};
+/* position of pnode (which is nasid>>1): */
+static int                     uv_nshift __read_mostly;
 
-DEFINE_PER_CPU(struct ptc_stats, ptcstats);
-DEFINE_PER_CPU(struct bau_control, bau_control);
+static unsigned long           uv_mmask __read_mostly;
+
+static DEFINE_PER_CPU(struct ptc_stats, ptcstats);
+static DEFINE_PER_CPU(struct bau_control, bau_control);
 
 /*
  * Free a software acknowledge hardware resource by clearing its Pending
@@ -44,8 +41,8 @@ DEFINE_PER_CPU(struct bau_control, bau_control);
  * be sent (the hardware will only do one reply per message).
  */
 static void uv_reply_to_message(int resource,
-                   struct bau_payload_queue_entry *msg,
-                   struct bau_msg_status *msp)
+                               struct bau_payload_queue_entry *msg,
+                               struct bau_msg_status *msp)
 {
        unsigned long dw;
 
@@ -55,7 +52,6 @@ static void uv_reply_to_message(int resource,
        if (msp)
                msp->seen_by.bits = 0;
        uv_write_local_mmr(UVH_LB_BAU_INTD_SOFTWARE_ACKNOWLEDGE_ALIAS, dw);
-       return;
 }
 
 /*
@@ -63,17 +59,17 @@ static void uv_reply_to_message(int resource,
  * Other cpu's may come here at the same time for this message.
  */
 static void uv_bau_process_message(struct bau_payload_queue_entry *msg,
-                      int msg_slot, int sw_ack_slot)
+                                  int msg_slot, int sw_ack_slot)
 {
-       int cpu;
        unsigned long this_cpu_mask;
        struct bau_msg_status *msp;
+       int cpu;
 
        msp = __get_cpu_var(bau_control).msg_statuses + msg_slot;
        cpu = uv_blade_processor_id();
        msg->number_of_cpus =
            uv_blade_nr_online_cpus(uv_node_to_blade_id(numa_node_id()));
-       this_cpu_mask = (unsigned long)1 << cpu;
+       this_cpu_mask = 1UL << cpu;
        if (msp->seen_by.bits & this_cpu_mask)
                return;
        atomic_or_long(&msp->seen_by.bits, this_cpu_mask);
@@ -94,7 +90,40 @@ static void uv_bau_process_message(struct bau_payload_queue_entry *msg,
        atomic_inc_short(&msg->acknowledge_count);
        if (msg->number_of_cpus == msg->acknowledge_count)
                uv_reply_to_message(sw_ack_slot, msg, msp);
-       return;
+}
+
+/*
+ * Examine the payload queue on one distribution node to see
+ * which messages have not been seen, and which cpu(s) have not seen them.
+ *
+ * Returns the number of cpu's that have not responded.
+ */
+static int uv_examine_destination(struct bau_control *bau_tablesp, int sender)
+{
+       struct bau_payload_queue_entry *msg;
+       struct bau_msg_status *msp;
+       int count = 0;
+       int i;
+       int j;
+
+       for (msg = bau_tablesp->va_queue_first, i = 0; i < DEST_Q_SIZE;
+            msg++, i++) {
+               if ((msg->sending_cpu == sender) && (!msg->replied_to)) {
+                       msp = bau_tablesp->msg_statuses + i;
+                       printk(KERN_DEBUG
+                              "blade %d: address:%#lx %d of %d, not cpu(s): ",
+                              i, msg->address, msg->acknowledge_count,
+                              msg->number_of_cpus);
+                       for (j = 0; j < msg->number_of_cpus; j++) {
+                               if (!((1L << j) & msp->seen_by.bits)) {
+                                       count++;
+                                       printk("%d ", j);
+                               }
+                       }
+                       printk("\n");
+               }
+       }
+       return count;
 }
 
 /*
@@ -107,40 +136,13 @@ static int uv_examine_destinations(struct bau_target_nodemask *distribution)
 {
        int sender;
        int i;
-       int j;
-       int k;
        int count = 0;
-       struct bau_control *bau_tablesp;
-       struct bau_payload_queue_entry *msg;
-       struct bau_msg_status *msp;
 
        sender = smp_processor_id();
-       for (i = 0; i < (sizeof(struct bau_target_nodemask) * BITSPERBYTE);
-            i++) {
+       for (i = 0; i < sizeof(struct bau_target_nodemask) * BITSPERBYTE; i++) {
                if (!bau_node_isset(i, distribution))
                        continue;
-               bau_tablesp = uv_bau_table_bases[i];
-               for (msg = bau_tablesp->va_queue_first, j = 0;
-                    j < DESTINATION_PAYLOAD_QUEUE_SIZE; msg++, j++) {
-                       if ((msg->sending_cpu == sender) &&
-                           (!msg->replied_to)) {
-                               msp = bau_tablesp->msg_statuses + j;
-                               printk(KERN_DEBUG
-                               "blade %d: address:%#lx %d of %d, not cpu(s): ",
-                                      i, msg->address,
-                                      msg->acknowledge_count,
-                                      msg->number_of_cpus);
-                               for (k = 0; k < msg->number_of_cpus;
-                                    k++) {
-                                       if (!((long)1 << k & msp->
-                                             seen_by.bits)) {
-                                               count++;
-                                               printk("%d ", k);
-                                       }
-                               }
-                               printk("\n");
-                       }
-               }
+               count += uv_examine_destination(uv_bau_table_bases[i], sender);
        }
        return count;
 }
@@ -150,7 +152,7 @@ static int uv_examine_destinations(struct bau_target_nodemask *distribution)
  *
  * return COMPLETE, RETRY or GIVEUP
  */
-static int uv_wait_completion(struct bau_activation_descriptor *bau_desc,
+static int uv_wait_completion(struct bau_desc *bau_desc,
                              unsigned long mmr_offset, int right_shift)
 {
        int exams = 0;
@@ -213,16 +215,16 @@ static int uv_wait_completion(struct bau_activation_descriptor *bau_desc,
  * Returns 0 if some remote flushing remains to be done. The mask is left
  * unchanged.
  */
-int uv_flush_send_and_wait(int cpu, int this_blade,
-       struct bau_activation_descriptor *bau_desc, cpumask_t *cpumaskp)
+int uv_flush_send_and_wait(int cpu, int this_blade, struct bau_desc *bau_desc,
+                          cpumask_t *cpumaskp)
 {
        int completion_status = 0;
        int right_shift;
-       int bit;
-       int blade;
        int tries = 0;
-       unsigned long index;
+       int blade;
+       int bit;
        unsigned long mmr_offset;
+       unsigned long index;
        cycles_t time1;
        cycles_t time2;
 
@@ -237,8 +239,8 @@ int uv_flush_send_and_wait(int cpu, int this_blade,
        time1 = get_cycles();
        do {
                tries++;
-               index = ((unsigned long)
-                       1 << UVH_LB_BAU_SB_ACTIVATION_CONTROL_PUSH_SHFT) | cpu;
+               index = (1UL << UVH_LB_BAU_SB_ACTIVATION_CONTROL_PUSH_SHFT) |
+                       cpu;
                uv_write_local_mmr(UVH_LB_BAU_SB_ACTIVATION_CONTROL, index);
                completion_status = uv_wait_completion(bau_desc, mmr_offset,
                                        right_shift);
@@ -295,7 +297,7 @@ int uv_flush_send_and_wait(int cpu, int this_blade,
  * Returns 0 if some remote flushing remains to be done.
  */
 int uv_flush_tlb_others(cpumask_t *cpumaskp, struct mm_struct *mm,
-       unsigned long va)
+                       unsigned long va)
 {
        int i;
        int bit;
@@ -303,7 +305,7 @@ int uv_flush_tlb_others(cpumask_t *cpumaskp, struct mm_struct *mm,
        int cpu;
        int this_blade;
        int locals = 0;
-       struct bau_activation_descriptor *bau_desc;
+       struct bau_desc *bau_desc;
 
        cpu = uv_blade_processor_id();
        this_blade = uv_numa_blade_id();
@@ -315,8 +317,7 @@ int uv_flush_tlb_others(cpumask_t *cpumaskp, struct mm_struct *mm,
        i = 0;
        for_each_cpu_mask(bit, *cpumaskp) {
                blade = uv_cpu_to_blade_id(bit);
-               if (blade > (UV_DISTRIBUTION_SIZE - 1))
-                       BUG();
+               BUG_ON(blade > (UV_DISTRIBUTION_SIZE - 1));
                if (blade == this_blade) {
                        locals++;
                        continue;
@@ -358,10 +359,12 @@ int uv_flush_tlb_others(cpumask_t *cpumaskp, struct mm_struct *mm,
  */
 void uv_bau_message_interrupt(struct pt_regs *regs)
 {
-       struct bau_payload_queue_entry *pqp;
+       struct bau_payload_queue_entry *va_queue_first;
+       struct bau_payload_queue_entry *va_queue_last;
        struct bau_payload_queue_entry *msg;
        struct pt_regs *old_regs = set_irq_regs(regs);
-       cycles_t time1, time2;
+       cycles_t time1;
+       cycles_t time2;
        int msg_slot;
        int sw_ack_slot;
        int fw;
@@ -376,19 +379,21 @@ void uv_bau_message_interrupt(struct pt_regs *regs)
 
        local_pnode = uv_blade_to_pnode(uv_numa_blade_id());
 
-       pqp = __get_cpu_var(bau_control).va_queue_first;
+       va_queue_first = __get_cpu_var(bau_control).va_queue_first;
+       va_queue_last = __get_cpu_var(bau_control).va_queue_last;
+
        msg = __get_cpu_var(bau_control).bau_msg_head;
        while (msg->sw_ack_vector) {
                count++;
                fw = msg->sw_ack_vector;
-               msg_slot = msg - pqp;
+               msg_slot = msg - va_queue_first;
                sw_ack_slot = ffs(fw) - 1;
 
                uv_bau_process_message(msg, msg_slot, sw_ack_slot);
 
                msg++;
-               if (msg > __get_cpu_var(bau_control).va_queue_last)
-                       msg = __get_cpu_var(bau_control).va_queue_first;
+               if (msg > va_queue_last)
+                       msg = va_queue_first;
                __get_cpu_var(bau_control).bau_msg_head = msg;
        }
        if (!count)
@@ -401,7 +406,6 @@ void uv_bau_message_interrupt(struct pt_regs *regs)
 
        irq_exit();
        set_irq_regs(old_regs);
-       return;
 }
 
 static void uv_enable_timeouts(void)
@@ -423,7 +427,6 @@ static void uv_enable_timeouts(void)
                pnode = uv_blade_to_pnode(blade);
                cur_cpu += uv_blade_nr_possible_cpus(i);
        }
-       return;
 }
 
 static void *uv_ptc_seq_start(struct seq_file *file, loff_t *offset)
@@ -485,11 +488,13 @@ static int uv_ptc_seq_show(struct seq_file *file, void *data)
  * >0: retry limit
  */
 static ssize_t uv_ptc_proc_write(struct file *file, const char __user *user,
-                 size_t count, loff_t *data)
+                                size_t count, loff_t *data)
 {
        long newmode;
        char optstr[64];
 
+       if (count == 0 || count > sizeof(optstr))
+               return -EINVAL;
        if (copy_from_user(optstr, user, count))
                return -EFAULT;
        optstr[count - 1] = '\0';
@@ -535,10 +540,10 @@ static ssize_t uv_ptc_proc_write(struct file *file, const char __user *user,
 }
 
 static const struct seq_operations uv_ptc_seq_ops = {
-       .start  = uv_ptc_seq_start,
-       .next   = uv_ptc_seq_next,
-       .stop   = uv_ptc_seq_stop,
-       .show   = uv_ptc_seq_show
+       .start          = uv_ptc_seq_start,
+       .next           = uv_ptc_seq_next,
+       .stop           = uv_ptc_seq_stop,
+       .show           = uv_ptc_seq_show
 };
 
 static int uv_ptc_proc_open(struct inode *inode, struct file *file)
@@ -568,6 +573,7 @@ static int __init uv_ptc_init(void)
        if (!proc_uv_ptc) {
                printk(KERN_ERR "unable to create %s proc entry\n",
                       UV_PTC_BASENAME);
+               remove_proc_entry("sgi_uv", NULL);
                return -EINVAL;
        }
        proc_uv_ptc->proc_fops = &proc_uv_ptc_operations;
@@ -582,61 +588,60 @@ static struct bau_control * __init uv_table_bases_init(int blade, int node)
        int i;
        int *ip;
        struct bau_msg_status *msp;
-       struct bau_control *bau_tablesp;
+       struct bau_control *bau_tabp;
 
-       bau_tablesp =
+       bau_tabp =
            kmalloc_node(sizeof(struct bau_control), GFP_KERNEL, node);
-       if (!bau_tablesp)
-               BUG();
-       bau_tablesp->msg_statuses =
+       BUG_ON(!bau_tabp);
+
+       bau_tabp->msg_statuses =
            kmalloc_node(sizeof(struct bau_msg_status) *
-                        DESTINATION_PAYLOAD_QUEUE_SIZE, GFP_KERNEL, node);
-       if (!bau_tablesp->msg_statuses)
-               BUG();
-       for (i = 0, msp = bau_tablesp->msg_statuses;
-            i < DESTINATION_PAYLOAD_QUEUE_SIZE; i++, msp++) {
+                        DEST_Q_SIZE, GFP_KERNEL, node);
+       BUG_ON(!bau_tabp->msg_statuses);
+
+       for (i = 0, msp = bau_tabp->msg_statuses; i < DEST_Q_SIZE; i++, msp++)
                bau_cpubits_clear(&msp->seen_by, (int)
                                  uv_blade_nr_possible_cpus(blade));
-       }
-       bau_tablesp->watching =
-           kmalloc_node(sizeof(int) * DESTINATION_NUM_RESOURCES,
-                        GFP_KERNEL, node);
-       if (!bau_tablesp->watching)
-               BUG();
-       for (i = 0, ip = bau_tablesp->watching;
-            i < DESTINATION_PAYLOAD_QUEUE_SIZE; i++, ip++) {
+
+       bau_tabp->watching =
+           kmalloc_node(sizeof(int) * DEST_NUM_RESOURCES, GFP_KERNEL, node);
+       BUG_ON(!bau_tabp->watching);
+
+       for (i = 0, ip = bau_tabp->watching; i < DEST_Q_SIZE; i++, ip++)
                *ip = 0;
-       }
-       uv_bau_table_bases[blade] = bau_tablesp;
-       return bau_tablesp;
+
+       uv_bau_table_bases[blade] = bau_tabp;
+
+       return bau_tabp;
 }
 
 /*
  * finish the initialization of the per-blade control structures
  */
-static void __init uv_table_bases_finish(int blade, int node, int cur_cpu,
-                                 struct bau_control *bau_tablesp,
-                                 struct bau_activation_descriptor *adp)
+static void __init
+uv_table_bases_finish(int blade, int node, int cur_cpu,
+                     struct bau_control *bau_tablesp,
+                     struct bau_desc *adp)
 {
-       int i;
        struct bau_control *bcp;
+       int i;
 
-       for (i = cur_cpu; i < (cur_cpu + uv_blade_nr_possible_cpus(blade));
-            i++) {
+       for (i = cur_cpu; i < cur_cpu + uv_blade_nr_possible_cpus(blade); i++) {
                bcp = (struct bau_control *)&per_cpu(bau_control, i);
-               bcp->bau_msg_head = bau_tablesp->va_queue_first;
-               bcp->va_queue_first = bau_tablesp->va_queue_first;
-               bcp->va_queue_last = bau_tablesp->va_queue_last;
-               bcp->watching = bau_tablesp->watching;
-               bcp->msg_statuses = bau_tablesp->msg_statuses;
-               bcp->descriptor_base = adp;
+
+               bcp->bau_msg_head       = bau_tablesp->va_queue_first;
+               bcp->va_queue_first     = bau_tablesp->va_queue_first;
+               bcp->va_queue_last      = bau_tablesp->va_queue_last;
+               bcp->watching           = bau_tablesp->watching;
+               bcp->msg_statuses       = bau_tablesp->msg_statuses;
+               bcp->descriptor_base    = adp;
        }
 }
 
 /*
  * initialize the sending side's sending buffers
  */
-static struct bau_activation_descriptor * __init
+static struct bau_desc * __init
 uv_activation_descriptor_init(int node, int pnode)
 {
        int i;
@@ -644,23 +649,26 @@ uv_activation_descriptor_init(int node, int pnode)
        unsigned long m;
        unsigned long n;
        unsigned long mmr_image;
-       struct bau_activation_descriptor *adp;
-       struct bau_activation_descriptor *ad2;
+       struct bau_desc *adp;
+       struct bau_desc *ad2;
 
-       adp = (struct bau_activation_descriptor *)
+       adp = (struct bau_desc *)
            kmalloc_node(16384, GFP_KERNEL, node);
-       if (!adp)
-               BUG();
+       BUG_ON(!adp);
+
        pa = __pa((unsigned long)adp);
        n = pa >> uv_nshift;
        m = pa & uv_mmask;
+
        mmr_image = uv_read_global_mmr64(pnode, UVH_LB_BAU_SB_DESCRIPTOR_BASE);
-       if (mmr_image)
+       if (mmr_image) {
                uv_write_global_mmr64(pnode, (unsigned long)
                                      UVH_LB_BAU_SB_DESCRIPTOR_BASE,
                                      (n << UV_DESC_BASE_PNODE_SHIFT | m));
+       }
+
        for (i = 0, ad2 = adp; i < UV_ACTIVATION_DESCRIPTOR_SIZE; i++, ad2++) {
-               memset(ad2, 0, sizeof(struct bau_activation_descriptor));
+               memset(ad2, 0, sizeof(struct bau_desc));
                ad2->header.sw_ack_flag = 1;
                ad2->header.base_dest_nodeid =
                    uv_blade_to_pnode(uv_cpu_to_blade_id(0));
@@ -677,18 +685,17 @@ uv_activation_descriptor_init(int node, int pnode)
 /*
  * initialize the destination side's receiving buffers
  */
-static struct bau_payload_queue_entry * __init uv_payload_queue_init(int node,
-                               int pnode, struct bau_control *bau_tablesp)
+static struct bau_payload_queue_entry * __init
+uv_payload_queue_init(int node, int pnode, struct bau_control *bau_tablesp)
 {
-       char *cp;
        struct bau_payload_queue_entry *pqp;
+       char *cp;
+
+       pqp = (struct bau_payload_queue_entry *) kmalloc_node(
+               (DEST_Q_SIZE + 1) * sizeof(struct bau_payload_queue_entry),
+               GFP_KERNEL, node);
+       BUG_ON(!pqp);
 
-       pqp = (struct bau_payload_queue_entry *)
-           kmalloc_node((DESTINATION_PAYLOAD_QUEUE_SIZE + 1) *
-                        sizeof(struct bau_payload_queue_entry),
-                        GFP_KERNEL, node);
-       if (!pqp)
-               BUG();
        cp = (char *)pqp + 31;
        pqp = (struct bau_payload_queue_entry *)(((unsigned long)cp >> 5) << 5);
        bau_tablesp->va_queue_first = pqp;
@@ -699,13 +706,12 @@ static struct bau_payload_queue_entry * __init uv_payload_queue_init(int node,
                              uv_physnodeaddr(pqp));
        uv_write_global_mmr64(pnode, UVH_LB_BAU_INTD_PAYLOAD_QUEUE_TAIL,
                              uv_physnodeaddr(pqp));
-       bau_tablesp->va_queue_last =
-           pqp + (DESTINATION_PAYLOAD_QUEUE_SIZE - 1);
+       bau_tablesp->va_queue_last = pqp + (DEST_Q_SIZE - 1);
        uv_write_global_mmr64(pnode, UVH_LB_BAU_INTD_PAYLOAD_QUEUE_LAST,
                              (unsigned long)
                              uv_physnodeaddr(bau_tablesp->va_queue_last));
-       memset(pqp, 0, sizeof(struct bau_payload_queue_entry) *
-              DESTINATION_PAYLOAD_QUEUE_SIZE);
+       memset(pqp, 0, sizeof(struct bau_payload_queue_entry) * DEST_Q_SIZE);
+
        return pqp;
 }
 
@@ -717,7 +723,7 @@ static int __init uv_init_blade(int blade, int node, int cur_cpu)
        int pnode;
        unsigned long pa;
        unsigned long apicid;
-       struct bau_activation_descriptor *adp;
+       struct bau_desc *adp;
        struct bau_payload_queue_entry *pqp;
        struct bau_control *bau_tablesp;
 
@@ -755,7 +761,7 @@ static int __init uv_bau_init(void)
 
        uv_bau_retry_limit = 1;
        uv_nshift = uv_hub_info->n_val;
-       uv_mmask = ((unsigned long)1 << uv_hub_info->n_val) - 1;
+       uv_mmask = (1UL << uv_hub_info->n_val) - 1;
        nblades = 0;
        last_blade = -1;
        for_each_online_node(node) {
@@ -767,8 +773,8 @@ static int __init uv_bau_init(void)
        }
        uv_bau_table_bases = (struct bau_control **)
            kmalloc(nblades * sizeof(struct bau_control *), GFP_KERNEL);
-       if (!uv_bau_table_bases)
-               BUG();
+       BUG_ON(!uv_bau_table_bases);
+
        last_blade = -1;
        for_each_online_node(node) {
                blade = uv_node_to_blade_id(node);
@@ -778,8 +784,9 @@ static int __init uv_bau_init(void)
                uv_init_blade(blade, node, cur_cpu);
                cur_cpu += uv_blade_nr_possible_cpus(blade);
        }
-       set_intr_gate(UV_BAU_MESSAGE, uv_bau_message_intr1);
+       alloc_intr_gate(UV_BAU_MESSAGE, uv_bau_message_intr1);
        uv_enable_timeouts();
+
        return 0;
 }
 __initcall(uv_bau_init);