diff options
| author | Paul Mundt <lethal@linux-sh.org> | 2011-01-26 04:23:27 -0500 |
|---|---|---|
| committer | Paul Mundt <lethal@linux-sh.org> | 2011-01-26 04:23:27 -0500 |
| commit | 6b620478e58677bb3f4ec884abb29ef9d68c7821 (patch) | |
| tree | 7a438fc5412587f9bd9b907d4d423ea52a498006 /arch/powerpc/platforms/cell | |
| parent | 906b17dc089f7fa87e37a9cfe6ee185efc90e0da (diff) | |
| parent | 6fb1b304255efc5c4c93874ac8c066272e257e28 (diff) | |
Merge branch 'master' of git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux-2.6 into common/serial-rework
Diffstat (limited to 'arch/powerpc/platforms/cell')
| -rw-r--r-- | arch/powerpc/platforms/cell/cpufreq_spudemand.c | 20 | ||||
| -rw-r--r-- | arch/powerpc/platforms/cell/qpace_setup.c | 5 | ||||
| -rw-r--r-- | arch/powerpc/platforms/cell/spu_base.c | 70 | ||||
| -rw-r--r-- | arch/powerpc/platforms/cell/spufs/file.c | 27 |
4 files changed, 80 insertions, 42 deletions
diff --git a/arch/powerpc/platforms/cell/cpufreq_spudemand.c b/arch/powerpc/platforms/cell/cpufreq_spudemand.c index 968c1c0b4d5b..d809836bcf5f 100644 --- a/arch/powerpc/platforms/cell/cpufreq_spudemand.c +++ b/arch/powerpc/platforms/cell/cpufreq_spudemand.c | |||
| @@ -39,8 +39,6 @@ struct spu_gov_info_struct { | |||
| 39 | }; | 39 | }; |
| 40 | static DEFINE_PER_CPU(struct spu_gov_info_struct, spu_gov_info); | 40 | static DEFINE_PER_CPU(struct spu_gov_info_struct, spu_gov_info); |
| 41 | 41 | ||
| 42 | static struct workqueue_struct *kspugov_wq; | ||
| 43 | |||
| 44 | static int calc_freq(struct spu_gov_info_struct *info) | 42 | static int calc_freq(struct spu_gov_info_struct *info) |
| 45 | { | 43 | { |
| 46 | int cpu; | 44 | int cpu; |
| @@ -71,14 +69,14 @@ static void spu_gov_work(struct work_struct *work) | |||
| 71 | __cpufreq_driver_target(info->policy, target_freq, CPUFREQ_RELATION_H); | 69 | __cpufreq_driver_target(info->policy, target_freq, CPUFREQ_RELATION_H); |
| 72 | 70 | ||
| 73 | delay = usecs_to_jiffies(info->poll_int); | 71 | delay = usecs_to_jiffies(info->poll_int); |
| 74 | queue_delayed_work_on(info->policy->cpu, kspugov_wq, &info->work, delay); | 72 | schedule_delayed_work_on(info->policy->cpu, &info->work, delay); |
| 75 | } | 73 | } |
| 76 | 74 | ||
| 77 | static void spu_gov_init_work(struct spu_gov_info_struct *info) | 75 | static void spu_gov_init_work(struct spu_gov_info_struct *info) |
| 78 | { | 76 | { |
| 79 | int delay = usecs_to_jiffies(info->poll_int); | 77 | int delay = usecs_to_jiffies(info->poll_int); |
| 80 | INIT_DELAYED_WORK_DEFERRABLE(&info->work, spu_gov_work); | 78 | INIT_DELAYED_WORK_DEFERRABLE(&info->work, spu_gov_work); |
| 81 | queue_delayed_work_on(info->policy->cpu, kspugov_wq, &info->work, delay); | 79 | schedule_delayed_work_on(info->policy->cpu, &info->work, delay); |
| 82 | } | 80 | } |
| 83 | 81 | ||
| 84 | static void spu_gov_cancel_work(struct spu_gov_info_struct *info) | 82 | static void spu_gov_cancel_work(struct spu_gov_info_struct *info) |
| @@ -152,27 +150,15 @@ static int __init spu_gov_init(void) | |||
| 152 | { | 150 | { |
| 153 | int ret; | 151 | int ret; |
| 154 | 152 | ||
| 155 | kspugov_wq = create_workqueue("kspugov"); | ||
| 156 | if (!kspugov_wq) { | ||
| 157 | printk(KERN_ERR "creation of kspugov failed\n"); | ||
| 158 | ret = -EFAULT; | ||
| 159 | goto out; | ||
| 160 | } | ||
| 161 | |||
| 162 | ret = cpufreq_register_governor(&spu_governor); | 153 | ret = cpufreq_register_governor(&spu_governor); |
| 163 | if (ret) { | 154 | if (ret) |
| 164 | printk(KERN_ERR "registration of governor failed\n"); | 155 | printk(KERN_ERR "registration of governor failed\n"); |
| 165 | destroy_workqueue(kspugov_wq); | ||
| 166 | goto out; | ||
| 167 | } | ||
| 168 | out: | ||
| 169 | return ret; | 156 | return ret; |
| 170 | } | 157 | } |
| 171 | 158 | ||
| 172 | static void __exit spu_gov_exit(void) | 159 | static void __exit spu_gov_exit(void) |
| 173 | { | 160 | { |
| 174 | cpufreq_unregister_governor(&spu_governor); | 161 | cpufreq_unregister_governor(&spu_governor); |
| 175 | destroy_workqueue(kspugov_wq); | ||
| 176 | } | 162 | } |
| 177 | 163 | ||
| 178 | 164 | ||
diff --git a/arch/powerpc/platforms/cell/qpace_setup.c b/arch/powerpc/platforms/cell/qpace_setup.c index 1b5749042756..d31c594cfdf3 100644 --- a/arch/powerpc/platforms/cell/qpace_setup.c +++ b/arch/powerpc/platforms/cell/qpace_setup.c | |||
| @@ -145,9 +145,4 @@ define_machine(qpace) { | |||
| 145 | .calibrate_decr = generic_calibrate_decr, | 145 | .calibrate_decr = generic_calibrate_decr, |
| 146 | .progress = qpace_progress, | 146 | .progress = qpace_progress, |
| 147 | .init_IRQ = iic_init_IRQ, | 147 | .init_IRQ = iic_init_IRQ, |
| 148 | #ifdef CONFIG_KEXEC | ||
| 149 | .machine_kexec = default_machine_kexec, | ||
| 150 | .machine_kexec_prepare = default_machine_kexec_prepare, | ||
| 151 | .machine_crash_shutdown = default_machine_crash_shutdown, | ||
| 152 | #endif | ||
| 153 | }; | 148 | }; |
diff --git a/arch/powerpc/platforms/cell/spu_base.c b/arch/powerpc/platforms/cell/spu_base.c index 8547e86bfb42..acfaccea5f4f 100644 --- a/arch/powerpc/platforms/cell/spu_base.c +++ b/arch/powerpc/platforms/cell/spu_base.c | |||
| @@ -37,6 +37,7 @@ | |||
| 37 | #include <asm/spu_csa.h> | 37 | #include <asm/spu_csa.h> |
| 38 | #include <asm/xmon.h> | 38 | #include <asm/xmon.h> |
| 39 | #include <asm/prom.h> | 39 | #include <asm/prom.h> |
| 40 | #include <asm/kexec.h> | ||
| 40 | 41 | ||
| 41 | const struct spu_management_ops *spu_management_ops; | 42 | const struct spu_management_ops *spu_management_ops; |
| 42 | EXPORT_SYMBOL_GPL(spu_management_ops); | 43 | EXPORT_SYMBOL_GPL(spu_management_ops); |
| @@ -727,6 +728,75 @@ static ssize_t spu_stat_show(struct sys_device *sysdev, | |||
| 727 | 728 | ||
| 728 | static SYSDEV_ATTR(stat, 0644, spu_stat_show, NULL); | 729 | static SYSDEV_ATTR(stat, 0644, spu_stat_show, NULL); |
| 729 | 730 | ||
| 731 | #ifdef CONFIG_KEXEC | ||
| 732 | |||
| 733 | struct crash_spu_info { | ||
| 734 | struct spu *spu; | ||
| 735 | u32 saved_spu_runcntl_RW; | ||
| 736 | u32 saved_spu_status_R; | ||
| 737 | u32 saved_spu_npc_RW; | ||
| 738 | u64 saved_mfc_sr1_RW; | ||
| 739 | u64 saved_mfc_dar; | ||
| 740 | u64 saved_mfc_dsisr; | ||
| 741 | }; | ||
| 742 | |||
| 743 | #define CRASH_NUM_SPUS 16 /* Enough for current hardware */ | ||
| 744 | static struct crash_spu_info crash_spu_info[CRASH_NUM_SPUS]; | ||
| 745 | |||
| 746 | static void crash_kexec_stop_spus(void) | ||
| 747 | { | ||
| 748 | struct spu *spu; | ||
| 749 | int i; | ||
| 750 | u64 tmp; | ||
| 751 | |||
| 752 | for (i = 0; i < CRASH_NUM_SPUS; i++) { | ||
| 753 | if (!crash_spu_info[i].spu) | ||
| 754 | continue; | ||
| 755 | |||
| 756 | spu = crash_spu_info[i].spu; | ||
| 757 | |||
| 758 | crash_spu_info[i].saved_spu_runcntl_RW = | ||
| 759 | in_be32(&spu->problem->spu_runcntl_RW); | ||
| 760 | crash_spu_info[i].saved_spu_status_R = | ||
| 761 | in_be32(&spu->problem->spu_status_R); | ||
| 762 | crash_spu_info[i].saved_spu_npc_RW = | ||
| 763 | in_be32(&spu->problem->spu_npc_RW); | ||
| 764 | |||
| 765 | crash_spu_info[i].saved_mfc_dar = spu_mfc_dar_get(spu); | ||
| 766 | crash_spu_info[i].saved_mfc_dsisr = spu_mfc_dsisr_get(spu); | ||
| 767 | tmp = spu_mfc_sr1_get(spu); | ||
| 768 | crash_spu_info[i].saved_mfc_sr1_RW = tmp; | ||
| 769 | |||
| 770 | tmp &= ~MFC_STATE1_MASTER_RUN_CONTROL_MASK; | ||
| 771 | spu_mfc_sr1_set(spu, tmp); | ||
| 772 | |||
| 773 | __delay(200); | ||
| 774 | } | ||
| 775 | } | ||
| 776 | |||
| 777 | static void crash_register_spus(struct list_head *list) | ||
| 778 | { | ||
| 779 | struct spu *spu; | ||
| 780 | int ret; | ||
| 781 | |||
| 782 | list_for_each_entry(spu, list, full_list) { | ||
| 783 | if (WARN_ON(spu->number >= CRASH_NUM_SPUS)) | ||
| 784 | continue; | ||
| 785 | |||
| 786 | crash_spu_info[spu->number].spu = spu; | ||
| 787 | } | ||
| 788 | |||
| 789 | ret = crash_shutdown_register(&crash_kexec_stop_spus); | ||
| 790 | if (ret) | ||
| 791 | printk(KERN_ERR "Could not register SPU crash handler"); | ||
| 792 | } | ||
| 793 | |||
| 794 | #else | ||
| 795 | static inline void crash_register_spus(struct list_head *list) | ||
| 796 | { | ||
| 797 | } | ||
| 798 | #endif | ||
| 799 | |||
| 730 | static int __init init_spu_base(void) | 800 | static int __init init_spu_base(void) |
| 731 | { | 801 | { |
| 732 | int i, ret = 0; | 802 | int i, ret = 0; |
diff --git a/arch/powerpc/platforms/cell/spufs/file.c b/arch/powerpc/platforms/cell/spufs/file.c index 02f7b113a31b..3c7c3f82d842 100644 --- a/arch/powerpc/platforms/cell/spufs/file.c +++ b/arch/powerpc/platforms/cell/spufs/file.c | |||
| @@ -219,24 +219,17 @@ spufs_mem_write(struct file *file, const char __user *buffer, | |||
| 219 | loff_t pos = *ppos; | 219 | loff_t pos = *ppos; |
| 220 | int ret; | 220 | int ret; |
| 221 | 221 | ||
| 222 | if (pos < 0) | ||
| 223 | return -EINVAL; | ||
| 224 | if (pos > LS_SIZE) | 222 | if (pos > LS_SIZE) |
| 225 | return -EFBIG; | 223 | return -EFBIG; |
| 226 | if (size > LS_SIZE - pos) | ||
| 227 | size = LS_SIZE - pos; | ||
| 228 | 224 | ||
| 229 | ret = spu_acquire(ctx); | 225 | ret = spu_acquire(ctx); |
| 230 | if (ret) | 226 | if (ret) |
| 231 | return ret; | 227 | return ret; |
| 232 | 228 | ||
| 233 | local_store = ctx->ops->get_ls(ctx); | 229 | local_store = ctx->ops->get_ls(ctx); |
| 234 | ret = copy_from_user(local_store + pos, buffer, size); | 230 | size = simple_write_to_buffer(local_store, LS_SIZE, ppos, buffer, size); |
| 235 | spu_release(ctx); | 231 | spu_release(ctx); |
| 236 | 232 | ||
| 237 | if (ret) | ||
| 238 | return -EFAULT; | ||
| 239 | *ppos = pos + size; | ||
| 240 | return size; | 233 | return size; |
| 241 | } | 234 | } |
| 242 | 235 | ||
| @@ -574,18 +567,15 @@ spufs_regs_write(struct file *file, const char __user *buffer, | |||
| 574 | if (*pos >= sizeof(lscsa->gprs)) | 567 | if (*pos >= sizeof(lscsa->gprs)) |
| 575 | return -EFBIG; | 568 | return -EFBIG; |
| 576 | 569 | ||
| 577 | size = min_t(ssize_t, sizeof(lscsa->gprs) - *pos, size); | ||
| 578 | *pos += size; | ||
| 579 | |||
| 580 | ret = spu_acquire_saved(ctx); | 570 | ret = spu_acquire_saved(ctx); |
| 581 | if (ret) | 571 | if (ret) |
| 582 | return ret; | 572 | return ret; |
| 583 | 573 | ||
| 584 | ret = copy_from_user((char *)lscsa->gprs + *pos - size, | 574 | size = simple_write_to_buffer(lscsa->gprs, sizeof(lscsa->gprs), pos, |
| 585 | buffer, size) ? -EFAULT : size; | 575 | buffer, size); |
| 586 | 576 | ||
| 587 | spu_release_saved(ctx); | 577 | spu_release_saved(ctx); |
| 588 | return ret; | 578 | return size; |
| 589 | } | 579 | } |
| 590 | 580 | ||
| 591 | static const struct file_operations spufs_regs_fops = { | 581 | static const struct file_operations spufs_regs_fops = { |
| @@ -630,18 +620,15 @@ spufs_fpcr_write(struct file *file, const char __user * buffer, | |||
| 630 | if (*pos >= sizeof(lscsa->fpcr)) | 620 | if (*pos >= sizeof(lscsa->fpcr)) |
| 631 | return -EFBIG; | 621 | return -EFBIG; |
| 632 | 622 | ||
| 633 | size = min_t(ssize_t, sizeof(lscsa->fpcr) - *pos, size); | ||
| 634 | |||
| 635 | ret = spu_acquire_saved(ctx); | 623 | ret = spu_acquire_saved(ctx); |
| 636 | if (ret) | 624 | if (ret) |
| 637 | return ret; | 625 | return ret; |
| 638 | 626 | ||
| 639 | *pos += size; | 627 | size = simple_write_to_buffer(&lscsa->fpcr, sizeof(lscsa->fpcr), pos, |
| 640 | ret = copy_from_user((char *)&lscsa->fpcr + *pos - size, | 628 | buffer, size); |
| 641 | buffer, size) ? -EFAULT : size; | ||
| 642 | 629 | ||
| 643 | spu_release_saved(ctx); | 630 | spu_release_saved(ctx); |
| 644 | return ret; | 631 | return size; |
| 645 | } | 632 | } |
| 646 | 633 | ||
| 647 | static const struct file_operations spufs_fpcr_fops = { | 634 | static const struct file_operations spufs_fpcr_fops = { |
