sigset.c 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363
  1. #include <string.h>
  2. #include <stdlib.h>
  3. #include <grass/imagery.h>
  4. #include <grass/gis.h>
  5. static int gettag( FILE *, char *);
  6. static int get_nbands( FILE *, struct SigSet *);
  7. static int get_title( FILE *, struct SigSet *);
  8. static int get_class( FILE *, struct SigSet *);
  9. static int get_classnum( FILE *, struct ClassSig *);
  10. static int get_classtype( FILE *, struct ClassSig *);
  11. static int get_classtitle( FILE *, struct ClassSig *);
  12. static int get_subclass ( FILE *, struct SigSet *, struct ClassSig *);
  13. static int get_subclass_pi( FILE *, struct SubSig *);
  14. static int get_subclass_means ( FILE *, struct SubSig *, int );
  15. static int get_subclass_covar ( FILE *, struct SubSig *, int );
  16. static double **alloc_matrix (int rows, int cols)
  17. {
  18. double **m;
  19. int i;
  20. m = (double **) G_calloc (rows, sizeof(double *));
  21. m[0] = (double *) G_calloc (rows*cols, sizeof(double));
  22. for (i = 1; i < rows; i++)
  23. m[i] = m[i-1] + cols;
  24. return m;
  25. }
  26. int I_SigSetNClasses(struct SigSet *S)
  27. {
  28. int i, count;
  29. for (i = 0, count = 0; i < S->nclasses; i++)
  30. if (S->ClassSig[i].used)
  31. count++;
  32. return count;
  33. }
  34. struct ClassData *
  35. I_AllocClassData (
  36. struct SigSet *S,
  37. struct ClassSig *C,
  38. int npixels)
  39. {
  40. struct ClassData *Data;
  41. Data = &(C->ClassData);
  42. Data->npixels = npixels;
  43. Data->count = 0;
  44. Data->x = alloc_matrix (npixels, S->nbands);
  45. Data->p = alloc_matrix (npixels, C->nsubclasses);
  46. return Data;
  47. }
  48. int I_InitSigSet (struct SigSet *S)
  49. {
  50. S->nbands = 0;
  51. S->nclasses = 0;
  52. S->ClassSig = NULL;
  53. S->title = NULL;
  54. return 0;
  55. }
  56. int I_SigSetNBands(struct SigSet *S, int nbands)
  57. {
  58. S->nbands = nbands;
  59. return 0;
  60. }
  61. struct ClassSig *
  62. I_NewClassSig (struct SigSet *S)
  63. {
  64. struct ClassSig *Sp;
  65. if (S->nclasses == 0)
  66. S->ClassSig = (struct ClassSig *) G_malloc (sizeof(struct ClassSig));
  67. else
  68. S->ClassSig = (struct ClassSig *) G_realloc ((char *)S->ClassSig,
  69. sizeof(struct ClassSig) * (S->nclasses+1));
  70. Sp = &S->ClassSig[S->nclasses++];
  71. Sp->classnum = 0;
  72. Sp->nsubclasses = 0;
  73. Sp->used = 1;
  74. Sp->type = SIGNATURE_TYPE_MIXED;
  75. Sp->title = NULL;
  76. return Sp;
  77. }
  78. struct SubSig *
  79. I_NewSubSig( struct SigSet *S, struct ClassSig *C)
  80. {
  81. struct SubSig *Sp;
  82. int i;
  83. if (C->nsubclasses == 0)
  84. C->SubSig = (struct SubSig *) G_malloc (sizeof(struct SubSig));
  85. else
  86. C->SubSig = (struct SubSig *) G_realloc ((char *)C->SubSig,
  87. sizeof(struct SubSig) * (C->nsubclasses+1));
  88. Sp = &C->SubSig[C->nsubclasses++];
  89. Sp->used = 1;
  90. Sp->R = (double **) G_calloc (S->nbands, sizeof(double *));
  91. Sp->R[0] = (double *) G_calloc (S->nbands * S->nbands, sizeof(double));
  92. for (i = 1; i < S->nbands; i++)
  93. Sp->R[i] = Sp->R[i-1] + S->nbands;
  94. Sp->Rinv = (double **) G_calloc (S->nbands, sizeof(double *));
  95. Sp->Rinv[0] = (double *) G_calloc (S->nbands * S->nbands, sizeof(double));
  96. for (i = 1; i < S->nbands; i++)
  97. Sp->Rinv[i] = Sp->Rinv[i-1] + S->nbands;
  98. Sp->means = (double *) G_calloc (S->nbands, sizeof(double));
  99. Sp->N = 0;
  100. Sp->pi = 0;
  101. Sp->cnst = 0;
  102. return Sp;
  103. }
  104. #define eq(a,b) strcmp(a,b)==0
  105. int I_ReadSigSet(FILE *fd, struct SigSet *S)
  106. {
  107. char tag[256];
  108. I_InitSigSet (S);
  109. while (gettag(fd, tag))
  110. {
  111. if (eq (tag, "title:"))
  112. get_title(fd, S);
  113. if (eq (tag, "nbands:"))
  114. get_nbands(fd, S);
  115. if (eq (tag, "class:"))
  116. get_class (fd, S);
  117. }
  118. return 1; /* for now assume success */
  119. }
  120. static int gettag (FILE *fd, char *tag)
  121. {
  122. if(fscanf(fd, "%s", tag) != 1) return 0;
  123. G_strip (tag);
  124. return 1;
  125. }
  126. static int get_nbands( FILE *fd, struct SigSet *S)
  127. {
  128. fscanf (fd, "%d", &S->nbands);
  129. return 0;
  130. }
  131. static int get_title( FILE *fd, struct SigSet *S)
  132. {
  133. char title[1024];
  134. *title = 0;
  135. fscanf (fd, "%[^\n]", title);
  136. I_SetSigTitle(S, title);
  137. return 0;
  138. }
  139. static int get_class( FILE *fd, struct SigSet *S)
  140. {
  141. char tag[1024];
  142. struct ClassSig *C;
  143. C = I_NewClassSig(S);
  144. while (gettag(fd, tag))
  145. {
  146. if (eq(tag, "endclass:"))
  147. break;
  148. if (eq(tag, "classnum:"))
  149. get_classnum(fd, C);
  150. if (eq(tag, "classtype:"))
  151. get_classtype(fd, C);
  152. if (eq(tag, "classtitle:"))
  153. get_classtitle(fd, C);
  154. if (eq(tag, "subclass:"))
  155. get_subclass(fd, S, C);
  156. }
  157. return 0;
  158. }
  159. static int get_classnum( FILE *fd, struct ClassSig *C)
  160. {
  161. fscanf (fd, "%ld", &C->classnum);
  162. return 0;
  163. }
  164. static int get_classtype( FILE *fd, struct ClassSig *C)
  165. {
  166. fscanf (fd, "%d", &C->type);
  167. return 0;
  168. }
  169. static int get_classtitle( FILE *fd, struct ClassSig *C)
  170. {
  171. char title[1024];
  172. *title = 0;
  173. fscanf (fd, "%[^\n]", title);
  174. I_SetClassTitle (C, title);
  175. return 0;
  176. }
  177. static int get_subclass( FILE *fd, struct SigSet *S, struct ClassSig *C)
  178. {
  179. struct SubSig *Sp;
  180. char tag[1024];
  181. Sp = I_NewSubSig(S,C);
  182. while (gettag(fd, tag))
  183. {
  184. if (eq(tag, "endsubclass:"))
  185. break;
  186. if (eq(tag, "pi:"))
  187. get_subclass_pi(fd, Sp);
  188. if (eq(tag, "means:"))
  189. get_subclass_means (fd, Sp, S->nbands);
  190. if (eq(tag, "covar:"))
  191. get_subclass_covar (fd, Sp, S->nbands);
  192. }
  193. return 0;
  194. }
  195. static int
  196. get_subclass_pi(
  197. FILE *fd,
  198. struct SubSig *Sp)
  199. {
  200. fscanf (fd, "%lf", &Sp->pi);
  201. return 0;
  202. }
  203. static int
  204. get_subclass_means (
  205. FILE *fd,
  206. struct SubSig *Sp,
  207. int nbands)
  208. {
  209. int i;
  210. for (i = 0; i < nbands; i++)
  211. fscanf (fd, "%lf", &Sp->means[i]);
  212. return 0;
  213. }
  214. static int
  215. get_subclass_covar(
  216. FILE *fd,
  217. struct SubSig *Sp,
  218. int nbands)
  219. {
  220. int i,j;
  221. for (i = 0; i < nbands; i++)
  222. for (j = 0; j < nbands; j++)
  223. fscanf (fd, "%lf", &Sp->R[i][j]);
  224. return 0;
  225. }
  226. int I_SetSigTitle(
  227. struct SigSet *S,
  228. const char *title)
  229. {
  230. if (title == NULL) title = "";
  231. if (S->title)
  232. free (S->title);
  233. S->title = G_store(title);
  234. return 0;
  235. }
  236. const char *I_GetSigTitle(const struct SigSet *S)
  237. {
  238. if (S->title)
  239. return S->title;
  240. else
  241. return "";
  242. }
  243. int I_SetClassTitle(
  244. struct ClassSig *C,
  245. const char *title)
  246. {
  247. if (title == NULL) title = "";
  248. if (C->title)
  249. free (C->title);
  250. C->title = G_store (title);
  251. return 0;
  252. }
  253. const char *I_GetClassTitle (const struct ClassSig *C)
  254. {
  255. if (C->title)
  256. return C->title;
  257. else
  258. return "";
  259. }
  260. int I_WriteSigSet(
  261. FILE *fd,
  262. const struct SigSet *S)
  263. {
  264. const struct ClassSig *Cp;
  265. const struct SubSig *Sp;
  266. int i,j, b1, b2;
  267. fprintf (fd, "title: %s\n",I_GetSigTitle(S));
  268. fprintf (fd, "nbands: %d\n", S->nbands);
  269. for (i=0; i < S->nclasses; i++)
  270. {
  271. Cp = &S->ClassSig[i];
  272. if (!Cp->used) continue;
  273. if (Cp->nsubclasses <= 0) continue;
  274. fprintf (fd, "class:\n");
  275. fprintf (fd, " classnum: %ld\n", Cp->classnum);
  276. fprintf (fd, " classtitle: %s\n",I_GetClassTitle(Cp));
  277. fprintf (fd, " classtype: %d\n", Cp->type);
  278. for (j = 0; j < Cp->nsubclasses; j++)
  279. {
  280. Sp = &Cp->SubSig[j];
  281. fprintf (fd, " subclass:\n");
  282. fprintf (fd, " pi: %f\n", Sp->pi);
  283. fprintf (fd, " means:");
  284. for (b1 = 0; b1 < S->nbands; b1++)
  285. fprintf (fd, " %f", Sp->means[b1]);
  286. fprintf (fd, "\n");
  287. fprintf (fd, " covar:\n");
  288. for (b1 = 0; b1 < S->nbands; b1++)
  289. {
  290. fprintf (fd, " ");
  291. for (b2 = 0; b2 < S->nbands; b2++)
  292. fprintf (fd, " %f", Sp->R[b1][b2]);
  293. fprintf (fd, "\n");
  294. }
  295. fprintf (fd, " endsubclass:\n");
  296. }
  297. fprintf (fd, "endclass:\n");
  298. }
  299. return 0;
  300. }