cluster.c 2.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. /*
  2. * QEMU CPU cluster
  3. *
  4. * Copyright (c) 2018 GreenSocs SAS
  5. *
  6. * This program is free software; you can redistribute it and/or
  7. * modify it under the terms of the GNU General Public License
  8. * as published by the Free Software Foundation; either version 2
  9. * of the License, or (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program; if not, see
  18. * <http://www.gnu.org/licenses/gpl-2.0.html>
  19. */
  20. #include "qemu/osdep.h"
  21. #include "hw/core/cpu.h"
  22. #include "hw/cpu/cluster.h"
  23. #include "hw/qdev-properties.h"
  24. #include "qapi/error.h"
  25. static const Property cpu_cluster_properties[] = {
  26. DEFINE_PROP_UINT32("cluster-id", CPUClusterState, cluster_id, 0),
  27. };
  28. typedef struct CallbackData {
  29. CPUClusterState *cluster;
  30. int cpu_count;
  31. } CallbackData;
  32. static int add_cpu_to_cluster(Object *obj, void *opaque)
  33. {
  34. CallbackData *cbdata = opaque;
  35. CPUState *cpu = (CPUState *)object_dynamic_cast(obj, TYPE_CPU);
  36. if (cpu) {
  37. cpu->cluster_index = cbdata->cluster->cluster_id;
  38. cbdata->cpu_count++;
  39. }
  40. return 0;
  41. }
  42. static void cpu_cluster_realize(DeviceState *dev, Error **errp)
  43. {
  44. /* Iterate through all our CPU children and set their cluster_index */
  45. CPUClusterState *cluster = CPU_CLUSTER(dev);
  46. Object *cluster_obj = OBJECT(dev);
  47. CallbackData cbdata = {
  48. .cluster = cluster,
  49. .cpu_count = 0,
  50. };
  51. if (cluster->cluster_id >= MAX_CLUSTERS) {
  52. error_setg(errp, "cluster-id must be less than %d", MAX_CLUSTERS);
  53. return;
  54. }
  55. object_child_foreach_recursive(cluster_obj, add_cpu_to_cluster, &cbdata);
  56. /*
  57. * A cluster with no CPUs is a bug in the board/SoC code that created it;
  58. * if you hit this during development of new code, check that you have
  59. * created the CPUs and parented them into the cluster object before
  60. * realizing the cluster object.
  61. */
  62. assert(cbdata.cpu_count > 0);
  63. }
  64. static void cpu_cluster_class_init(ObjectClass *klass, void *data)
  65. {
  66. DeviceClass *dc = DEVICE_CLASS(klass);
  67. device_class_set_props(dc, cpu_cluster_properties);
  68. dc->realize = cpu_cluster_realize;
  69. /* This is not directly for users, CPU children must be attached by code */
  70. dc->user_creatable = false;
  71. }
  72. static const TypeInfo cpu_cluster_type_info = {
  73. .name = TYPE_CPU_CLUSTER,
  74. .parent = TYPE_DEVICE,
  75. .instance_size = sizeof(CPUClusterState),
  76. .class_init = cpu_cluster_class_init,
  77. };
  78. static void cpu_cluster_register_types(void)
  79. {
  80. type_register_static(&cpu_cluster_type_info);
  81. }
  82. type_init(cpu_cluster_register_types)