#ifdef CONFIG_HOTPLUG_SMT
 
+static bool cpu_smt_num_threads_valid(unsigned int threads)
+{
+       if (IS_ENABLED(CONFIG_SMT_NUM_THREADS_DYNAMIC))
+               return threads >= 1 && threads <= cpu_smt_max_threads;
+       return threads == 1 || threads == cpu_smt_max_threads;
+}
+
 static ssize_t
 __store_smt_control(struct device *dev, struct device_attribute *attr,
                    const char *buf, size_t count)
 {
-       int ctrlval, ret;
+       int ctrlval, ret, num_threads, orig_threads;
+       bool force_off;
 
        if (cpu_smt_control == CPU_SMT_FORCE_DISABLED)
                return -EPERM;
        if (cpu_smt_control == CPU_SMT_NOT_SUPPORTED)
                return -ENODEV;
 
-       if (sysfs_streq(buf, "on"))
+       if (sysfs_streq(buf, "on")) {
                ctrlval = CPU_SMT_ENABLED;
-       else if (sysfs_streq(buf, "off"))
+               num_threads = cpu_smt_max_threads;
+       } else if (sysfs_streq(buf, "off")) {
                ctrlval = CPU_SMT_DISABLED;
-       else if (sysfs_streq(buf, "forceoff"))
+               num_threads = 1;
+       } else if (sysfs_streq(buf, "forceoff")) {
                ctrlval = CPU_SMT_FORCE_DISABLED;
-       else
+               num_threads = 1;
+       } else if (kstrtoint(buf, 10, &num_threads) == 0) {
+               if (num_threads == 1)
+                       ctrlval = CPU_SMT_DISABLED;
+               else if (cpu_smt_num_threads_valid(num_threads))
+                       ctrlval = CPU_SMT_ENABLED;
+               else
+                       return -EINVAL;
+       } else {
                return -EINVAL;
+       }
 
        ret = lock_device_hotplug_sysfs();
        if (ret)
                return ret;
 
-       if (ctrlval != cpu_smt_control) {
-               switch (ctrlval) {
-               case CPU_SMT_ENABLED:
-                       ret = cpuhp_smt_enable();
-                       break;
-               case CPU_SMT_DISABLED:
-               case CPU_SMT_FORCE_DISABLED:
-                       ret = cpuhp_smt_disable(ctrlval);
-                       break;
-               }
-       }
+       orig_threads = cpu_smt_num_threads;
+       cpu_smt_num_threads = num_threads;
+
+       force_off = ctrlval != cpu_smt_control && ctrlval == CPU_SMT_FORCE_DISABLED;
+
+       if (num_threads > orig_threads)
+               ret = cpuhp_smt_enable();
+       else if (num_threads < orig_threads || force_off)
+               ret = cpuhp_smt_disable(ctrlval);
 
        unlock_device_hotplug();
        return ret ? ret : count;
 {
        const char *state = smt_states[cpu_smt_control];
 
+#ifdef CONFIG_HOTPLUG_SMT
+       /*
+        * If SMT is enabled but not all threads are enabled then show the
+        * number of threads. If all threads are enabled show "on". Otherwise
+        * show the state name.
+        */
+       if (cpu_smt_control == CPU_SMT_ENABLED &&
+           cpu_smt_num_threads != cpu_smt_max_threads)
+               return sysfs_emit(buf, "%d\n", cpu_smt_num_threads);
+#endif
+
        return snprintf(buf, PAGE_SIZE - 2, "%s\n", state);
 }