unit.c 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <strings.h>
  5. #include "pci.h"
  6. #include "pcilib.h"
  7. #include "unit.h"
  8. #include "error.h"
  9. static pcilib_unit_transform_t pcilib_unit_transform_null = { NULL, NULL };
  10. int pcilib_add_units(pcilib_t *ctx, size_t n, const pcilib_unit_description_t *desc) {
  11. size_t i;
  12. int err = 0;
  13. if (!n) {
  14. for (n = 0; desc[n].name; n++);
  15. }
  16. if ((ctx->num_units + n + 1) > ctx->alloc_units) {
  17. size_t size;
  18. pcilib_unit_description_t *units;
  19. for (size = ctx->alloc_units; size < 2 * (n + ctx->num_units + 1); size <<= 1);
  20. units = (pcilib_unit_description_t*)realloc(ctx->units, size * sizeof(pcilib_unit_description_t));
  21. if (!units) return PCILIB_ERROR_MEMORY;
  22. ctx->units = units;
  23. ctx->alloc_units = size;
  24. ctx->model_info.units = units;
  25. }
  26. // ToDo: Check if exists...
  27. for (i = 0; i < n; i++) {
  28. pcilib_unit_t unit = pcilib_find_unit_by_name(ctx, desc[i].name);
  29. if (unit != PCILIB_UNIT_INVALID) {
  30. pcilib_clean_units(ctx, ctx->num_units);
  31. pcilib_error("Unit %s is already defined in the model", desc[i].name);
  32. return PCILIB_ERROR_EXIST;
  33. }
  34. pcilib_unit_context_t *unit_ctx = (pcilib_unit_context_t*)malloc(sizeof(pcilib_unit_context_t));
  35. if (!unit_ctx) {
  36. pcilib_clean_units(ctx, ctx->num_units);
  37. return PCILIB_ERROR_MEMORY;
  38. }
  39. memset(unit_ctx, 0, sizeof(pcilib_unit_context_t));
  40. unit_ctx->unit = ctx->num_units + i;
  41. unit_ctx->name = desc[i].name;
  42. HASH_ADD_KEYPTR(hh, ctx->unit_hash, unit_ctx->name, strlen(unit_ctx->name), unit_ctx);
  43. memcpy(ctx->units + ctx->num_units + i, &desc[i], sizeof(pcilib_unit_description_t));
  44. }
  45. ctx->num_units += n;
  46. memset(ctx->units + ctx->num_units, 0, sizeof(pcilib_unit_description_t));
  47. return err;
  48. }
  49. void pcilib_clean_units(pcilib_t *ctx, pcilib_unit_t start) {
  50. pcilib_unit_context_t *s, *tmp;
  51. if (ctx->unit_hash) {
  52. HASH_ITER(hh, ctx->unit_hash, s, tmp) {
  53. if (s->unit >= start) {
  54. HASH_DEL(ctx->unit_hash, s);
  55. free(s);
  56. }
  57. }
  58. }
  59. memset(&ctx->units[start], 0, sizeof(pcilib_unit_description_t));
  60. ctx->num_units = start;
  61. }
  62. pcilib_unit_t pcilib_find_unit_by_name(pcilib_t *ctx, const char *name) {
  63. pcilib_unit_context_t *unit_ctx = NULL;
  64. HASH_FIND_STR(ctx->unit_hash, name, unit_ctx);
  65. if (unit_ctx) return unit_ctx->unit;
  66. /*
  67. pcilib_unit_t i;
  68. for(i = 0; ctx->units[i].name; i++) {
  69. if (!strcasecmp(ctx->units[i].name, name))
  70. return i;
  71. }
  72. */
  73. return PCILIB_UNIT_INVALID;
  74. }
  75. pcilib_unit_transform_t *pcilib_find_transform_by_unit_names(pcilib_t *ctx, const char *from, const char *to) {
  76. int i;
  77. pcilib_unit_t unit;
  78. if ((!from)||(!to))
  79. return NULL;
  80. if (!strcasecmp(from, to))
  81. return &pcilib_unit_transform_null;
  82. unit = pcilib_find_unit_by_name(ctx, from);
  83. if (unit == PCILIB_UNIT_INVALID) return NULL;
  84. for (i = 0; ctx->units[unit].transforms[i].unit; i++) {
  85. if (!strcasecmp(ctx->units[unit].transforms[i].unit, to))
  86. return &ctx->units[unit].transforms[i];
  87. }
  88. return NULL;
  89. }
  90. int pcilib_transform_unit(pcilib_t *ctx, const pcilib_unit_transform_t *trans, pcilib_value_t *value) {
  91. int err;
  92. if (trans->transform) {
  93. err = pcilib_py_eval_string(ctx, trans->transform, value);
  94. if (err) return err;
  95. value->unit = trans->unit;
  96. } else if (trans->unit) {
  97. value->unit = trans->unit;
  98. }
  99. return 0;
  100. }
  101. int pcilib_transform_unit_by_name(pcilib_t *ctx, const char *to, pcilib_value_t *value) {
  102. pcilib_unit_transform_t *trans;
  103. if (!value->unit) {
  104. pcilib_warning("Can't transform unit of the value with unspecified unit");
  105. return PCILIB_ERROR_INVALID_ARGUMENT;
  106. }
  107. trans = pcilib_find_transform_by_unit_names(ctx, value->unit, to);
  108. if (!trans) {
  109. pcilib_warning("Can't transform unit (%s) to (%s)", value->unit, to);
  110. return PCILIB_ERROR_NOTSUPPORTED;
  111. }
  112. return pcilib_transform_unit(ctx, trans, value);
  113. }