// cs160_861.c
// http://i...content-available-to-author-only...e.com/gN66JU
#include <stdio.h>
#include <stdlib.h>
// [名前]
// cs160_861 : RGB → HSV
// [機能]
// RGBの3枚のグレースケールから
// HSVの3枚のグレースケールを出力する
// [資料]
// http://c...content-available-to-author-only...d.org/oNUxByE7
// http://j...content-available-to-author-only...a.org/wiki/HSV%E8%89%B2%E7%A9%BA%E9%96%93#RGB.E3.81.8B.E3.82.89HSV.E3.81.B8.E3.81.AE.E5.A4.89.E6.8F.9B
// [コマンドオプション]
// argv[1] : 幅
// argv[2] : 高さ
// argv[3] : 入力 R ファイル名
// argv[4] : 入力 G ファイル名
// argv[5] : 入力 B ファイル名
// argv[6] : 出力Hファイル名
// argv[7] : 出力Sファイル名
// argv[8] : 出力Vファイル名
// [入出力ファイルの仕様]
// 入力RGBファイル
// ・R,G,B それぞれ1ファイルである
// ・HEADER 512 bytes
// ・1dotにつき、百分率16bitリトルエンディアン (0~65535 [100%(65536)は存在しないこととする])
// 出力HSVファイル
// ・H,S,V それぞれ1ファイルである
// ・HEADER 無し
// ・1dotにつき、百分率16bitリトルエンディアン (0~65535 [65535が100%である])
// H (0.0以上 360.0以下) ~ H値 / 360.0 * 65535
// S (0.0以上 1.0以下) ~ S値 * 65535
// V (0.0以上 1.0以下) ~ V値 * 65535
// ※未定義の場合は0x0000をセットする
// 定数
const int header_size_rgb_file = 512; // RGBファイルのヘッダサイズ
const int comvert_mode = 1; // 変換モード (1:円柱モデル変換, 2:円錐モデル変換)
// fopen(); のモード
#ifdef _WIN32
// Windows
#define FOPEN_MODE_R "rb"
#define FOPEN_MODE_W "wb"
#else
// Linux
#define FOPEN_MODE_R "r"
#define FOPEN_MODE_W "w"
#endif
enum ERR_CODE // エラーコード
{
ERR_Normal = 0 , // 正常終了
ERR_Option , // 起動オプションが足りない
ERR_Width , // 入力幅が1未満
ERR_Height , // 入力高が1未満
ERR_InFile_Size , // 入力ファイルサイズが変である
// 以下6個は並び順を変えないこと →→→→→→→→
ERR_InFile_R , // 入力Rファイルオープンエラー
ERR_InFile_G , // 入力Gファイルオープンエラー
ERR_InFile_B , // 入力Bファイルオープンエラー
ERR_OutFile_H , // 出力Hファイルオープンエラー
ERR_OutFile_S , // 出力Sファイルオープンエラー
ERR_OutFile_V , // 出力Vファイルオープンエラー
// ←←←←←←←←←←←←←←←←←←←←←←←
ERR_MemAlloc , // メモリ確保エラー
ERR_Cnv_mode , // cnv_rgb2hsv();でcomvert_modeに不正な値がセットされていた(Prog Error)
};
// 中でやり取りするデータ
typedef struct
{
// 幅、高さ
int width, height;
int img_size;
// 入出力 ファイル
FILE * fp_in[3];
FILE * fp_out[3];
// データ
unsigned short *src[3]; // [0]R, [1]G, [2]B
unsigned short *dst[3]; // [0]H, [1]S, [2]V
} RGB_HSV_DATA;
// 関数
extern void load_rgb_data(RGB_HSV_DATA* data); // Load
extern int cnv_rgb2hsv(RGB_HSV_DATA* data); // Convert
extern int save_hsv_data(RGB_HSV_DATA* data); // Save
extern void get_max_min_mxcolor // max, min, maxのcolor を調べる
( /*[in]*/unsigned short R, unsigned short G, unsigned short B,
/*[out]*/unsigned short *mx, unsigned short *mn, char *mxcolor );
#define MIN(x, y) ((x)<(y) ? (x) : (y)) // 2値の最小値
#define MIN3(x, y, z) MIN(MIN((x), (y)), (z)) // 3値の最小値
// グローバル
char g_flagEndian; // 現在の環境のエンディアン (0:ビッグ / 1:リトル)
/****************************************************************************
* main
****************************************************************************/
int main(int argc, char *argv[])
{
int res; // 終了コード
RGB_HSV_DATA rh_data; // RGB,HSV データ
int i; // 汎用
/*
* 初期化 & ファイルオープン & チェック
*/
i = 1;g_flagEndian = *((char *)(&i)); // 現在の環境のエンディアン判定 (0:ビッグ / 1:リトル)
memset(&rh_data
, 0x00, sizeof(RGB_HSV_DATA
)); res = ERR_Normal;
if(argc != 9)
{
res = ERR_Option;
goto ERR_END;
}
// 幅
rh_data.
width = atoi (argv
[1]); if(rh_data.width < 0)
{
res = ERR_Width;
goto ERR_END;
}
// 高さ
rh_data.
height = atoi (argv
[2]); if(rh_data.height < 0)
{
res = ERR_Height;
goto ERR_END;
}
rh_data.img_size = rh_data.width * rh_data.height;
// ファイルオープン
for(i=0;i<3;i++)
{
if(NULL
== (rh_data.
fp_in[i
] = fopen(argv
[3+i
], FOPEN_MODE_R
))) {
res = ERR_InFile_R + i;
goto ERR_END;
}
}
for(i=0;i<3;i++)
{
if(NULL
==(rh_data.
fp_out[i
]=fopen(argv
[6+i
], FOPEN_MODE_W
))) {
res = ERR_OutFile_H + i;
goto ERR_END;
}
}
// ファイルサイズチェック
for(i=0;i<3;i++)
{
fseek(rh_data.
fp_in[i
], 0, SEEK_END
); if(ftell(rh_data.
fp_in[i
]) != rh_data.
img_size * sizeof(unsigned short) + header_size_rgb_file
) {
res = ERR_InFile_Size;
goto ERR_END;
}
}
/*
* メモリ確保
*/
for(i=0; i<3; i++)
{
if(NULL
== (rh_data.
src[i
] = (unsigned short*)malloc(rh_data.
img_size*sizeof(unsigned short)))) {
res = ERR_MemAlloc;
goto ERR_END;
}
if(NULL
== (rh_data.
dst[i
] = (unsigned short*)malloc(rh_data.
img_size*sizeof(unsigned short)))) {
res = ERR_MemAlloc;
goto ERR_END;
}
}
/*
* ロード
*/
load_rgb_data(&rh_data);
/*
* 変換
*/
res = cnv_rgb2hsv(&rh_data);
if(ERR_Normal != res)
{
goto ERR_END;
}
/*
* セーブ
*/
res = save_hsv_data(&rh_data);
if(ERR_Normal != res)
{
goto ERR_END;
}
/*
* 終了
*/
res = ERR_Normal;
ERR_END:;
// 解放
for(i=0;i<3;i++)
{
if(rh_data.
dst[i
]) { free(rh_data.
dst[i
]); rh_data.
dst[i
] = NULL
; } if(rh_data.
src[i
]) { free(rh_data.
src[i
]); rh_data.
src[i
] = NULL
; } if(rh_data.
fp_out[i
]) { fclose(rh_data.
fp_out[i
]); rh_data.
fp_out[i
] = NULL
; } if(rh_data.
fp_in [i
]) { fclose(rh_data.
fp_in [i
]); rh_data.
fp_in [i
] = NULL
; } }
return res;
}
// Load
void load_rgb_data(RGB_HSV_DATA* data)
{
int i, j;
unsigned short *p, v;
for(i=0; i<3; i++)
{
fseek(data
->fp_in
[i
], header_size_rgb_file
, SEEK_SET
); fread(data
->src
[i
], sizeof(unsigned short), data
->img_size
, data
->fp_in
[i
]); // 全部読む if(0 == g_flagEndian)
{
// big endian (入れ替える必要がある)
p = data->src[i];
for(j=0; j<data->img_size; j++)
{
v = *p;
*p++ = ((v >> 8) & 0xFF) | ((v << 8) & 0xFF00);
}
}
}
}
// Save
int save_hsv_data(RGB_HSV_DATA* data)
{
int i, j;
unsigned short *temp = NULL; // エンディアン変換バッファ
unsigned short *p, v;
// 確保
if(0 == g_flagEndian)
{
// big endian
if(NULL
== (temp
= (unsigned short*)malloc(data
->img_size
* sizeof(unsigned short)))) {
return ERR_MemAlloc;
}
}
// 書き込み
for(i=0; i<3; i++)
{
fseek(data
->fp_out
[i
],0,SEEK_SET
); if(0 == g_flagEndian)
{
memcpy(temp
, &(data
->dst
[i
][0]), sizeof(unsigned short) * data
->img_size
); p = temp;
for(j=0; j<data->img_size; j++)
{
v = *p;
*p++ = ((v >> 8) & 0xFF) | ((v << 8) & 0xFF00);
}
p = temp;
} else {
p = &(data->dst[i][0]);
}
fwrite(p
, sizeof(unsigned short), data
->img_size
, data
->fp_out
[i
]); }
// 解放
if(temp)
{
}
return 0;
}
// max, min, maxのcolor を調べる
// mxcolor = 'R', 'G', 'B'のいずれか
void get_max_min_mxcolor(/*[in]*/unsigned short R, unsigned short G, unsigned short B, /*[out]*/unsigned short *mx, unsigned short *mn, char *mxcolor)
{
// max
*mx = R; *mxcolor='R';
if(G > *mx) { *mx = G; *mxcolor='G'; }
if(B > *mx) { *mx = B; *mxcolor='B'; }
// min
*mn = MIN3(R, G, B);
}
// Convert
int cnv_rgb2hsv(RGB_HSV_DATA* data)
{
unsigned short max, min;
char color_max; // maxの色 ('R','G','B')
unsigned short H,S,V; // HSV
double th; // H計算用
double sub_maxmin; // H計算用 (max - min)
int i; // 汎用
for(i=0; i<data->img_size; i++)
{
// max, min, maxの色を取得
get_max_min_mxcolor(data->src[0][i], data->src[1][i], data->src[2][i], &max, &min, &color_max);
// SVを求める
// ( http://j...content-available-to-author-only...a.org/wiki/HSV%E8%89%B2%E7%A9%BA%E9%96%93#RGB.E3.81.8B.E3.82.89HSV.E3.81.B8.E3.81.AE.E5.A4.89.E6.8F.9B )
V = max;
if(max == min)
{
S = 0;
}else{
if(max == 0)
{
S = 0; // 未定義である
}else{
if(comvert_mode == 1)
{
S = (unsigned short)((double)(max - min) / (double)max * 65535.0);
}
else if(comvert_mode == 2)
{
S = max - min;
} else {
return ERR_Cnv_mode;
}
}
}
// Hを求める
if(max == min)
{
th = 0.0; // 未定義である
}else{
sub_maxmin = (double)max - (double)min;
switch(color_max)
{
case 'R':th = 60.0 * ((double)data->src[1][i] - (double)data->src[2][i]) / sub_maxmin + 0.0;break;
case 'G':th = 60.0 * ((double)data->src[2][i] - (double)data->src[0][i]) / sub_maxmin + 120.0;break;
case 'B':th = 60.0 * ((double)data->src[0][i] - (double)data->src[1][i]) / sub_maxmin + 240.0;break;
}
}
// 0.0以上360.0未満に丸める
while(th < 0.0)
{
th += 360.0;
}
while(th >= 360.0)
{
th -= 360.0;
}
H = (unsigned short)(th / 360.0 * 65535.0); // 360.0の百分率を16bit化
// HSVをセットする
data->dst[0][i] = H;
data->dst[1][i] = S;
data->dst[2][i] = V;
}
return ERR_Normal;
}
// End of cs160_861.c
//  cs160_861.c
//    http://i...content-available-to-author-only...e.com/gN66JU
#include <stdio.h>
#include <stdlib.h>

//  [名前]
//      cs160_861 : RGB → HSV
//  [機能]
//      RGBの3枚のグレースケールから
//      HSVの3枚のグレースケールを出力する
//  [資料]
//      http://c...content-available-to-author-only...d.org/oNUxByE7
//      http://j...content-available-to-author-only...a.org/wiki/HSV%E8%89%B2%E7%A9%BA%E9%96%93#RGB.E3.81.8B.E3.82.89HSV.E3.81.B8.E3.81.AE.E5.A4.89.E6.8F.9B
//  [コマンドオプション]
//      argv[1] : 幅
//      argv[2] : 高さ
//      argv[3] : 入力 R ファイル名
//      argv[4] : 入力 G ファイル名
//      argv[5] : 入力 B ファイル名
//      argv[6] : 出力Hファイル名
//      argv[7] : 出力Sファイル名
//      argv[8] : 出力Vファイル名
//  [入出力ファイルの仕様]
//      入力RGBファイル
//        ・R,G,B それぞれ1ファイルである
//        ・HEADER 512 bytes
//        ・1dotにつき、百分率16bitリトルエンディアン (0～65535 [100%(65536)は存在しないこととする])
//      出力HSVファイル
//        ・H,S,V それぞれ1ファイルである
//        ・HEADER 無し
//        ・1dotにつき、百分率16bitリトルエンディアン (0～65535 [65535が100%である])
//            H (0.0以上 360.0以下) ～ H値 / 360.0 * 65535
//            S (0.0以上   1.0以下) ～ S値         * 65535
//            V (0.0以上   1.0以下) ～ V値         * 65535
//          ※未定義の場合は0x0000をセットする


// 定数
const int header_size_rgb_file  = 512;  // RGBファイルのヘッダサイズ
const int comvert_mode          =   1;  // 変換モード (1:円柱モデル変換, 2:円錐モデル変換)
// fopen(); のモード
#ifdef _WIN32
    // Windows
    #define FOPEN_MODE_R  "rb"
    #define FOPEN_MODE_W  "wb"
#else
    // Linux
    #define FOPEN_MODE_R  "r"
    #define FOPEN_MODE_W  "w"
#endif

enum ERR_CODE                           // エラーコード
{
    ERR_Normal = 0  , // 正常終了
    ERR_Option      , // 起動オプションが足りない
    ERR_Width       , // 入力幅が1未満
    ERR_Height      , // 入力高が1未満
    ERR_InFile_Size , // 入力ファイルサイズが変である
    // 以下6個は並び順を変えないこと →→→→→→→→
    ERR_InFile_R    , // 入力Rファイルオープンエラー
    ERR_InFile_G    , // 入力Gファイルオープンエラー
    ERR_InFile_B    , // 入力Bファイルオープンエラー
    ERR_OutFile_H   , // 出力Hファイルオープンエラー
    ERR_OutFile_S   , // 出力Sファイルオープンエラー
    ERR_OutFile_V   , // 出力Vファイルオープンエラー
    // ←←←←←←←←←←←←←←←←←←←←←←←
    ERR_MemAlloc    , // メモリ確保エラー
    ERR_Cnv_mode    , // cnv_rgb2hsv();でcomvert_modeに不正な値がセットされていた(Prog Error)
};

// 中でやり取りするデータ
typedef struct
{
    // 幅、高さ
    int             width, height;
    int             img_size;
    // 入出力 ファイル
    FILE *          fp_in[3];
    FILE *          fp_out[3];
    // データ
    unsigned short  *src[3];    // [0]R, [1]G, [2]B
    unsigned short  *dst[3];    // [0]H, [1]S, [2]V
} RGB_HSV_DATA;


// 関数
extern void load_rgb_data(RGB_HSV_DATA* data);  // Load
extern int  cnv_rgb2hsv(RGB_HSV_DATA* data);    // Convert
extern int  save_hsv_data(RGB_HSV_DATA* data);  // Save
extern void get_max_min_mxcolor                 // max, min, maxのcolor を調べる
                ( /*[in]*/unsigned short R, unsigned short G, unsigned short B,
                  /*[out]*/unsigned short *mx, unsigned short *mn, char *mxcolor );
#define MIN(x, y)       ((x)<(y) ? (x) : (y))   // 2値の最小値
#define MIN3(x, y, z)   MIN(MIN((x), (y)), (z)) // 3値の最小値

// グローバル
char g_flagEndian;  // 現在の環境のエンディアン (0:ビッグ / 1:リトル)


/****************************************************************************
 *          main
 ****************************************************************************/
int main(int argc, char *argv[])
{
    int             res;        // 終了コード
    RGB_HSV_DATA    rh_data;    // RGB,HSV データ
    int             i;          // 汎用

    /*
     *   初期化 & ファイルオープン & チェック
     */
    i = 1;g_flagEndian = *((char *)(&i));       // 現在の環境のエンディアン判定 (0:ビッグ / 1:リトル)
    memset(&rh_data, 0x00, sizeof(RGB_HSV_DATA));
    res = ERR_Normal;
    if(argc != 9)
    {
        res = ERR_Option;
        goto ERR_END;
    }
    // 幅
    rh_data.width = atoi (argv[1]);
    if(rh_data.width < 0)
    {
        res = ERR_Width;
        goto ERR_END;
    }
    // 高さ
    rh_data.height = atoi (argv[2]);
    if(rh_data.height < 0)
    {
        res = ERR_Height;
        goto ERR_END;
    }
    rh_data.img_size = rh_data.width * rh_data.height;
    // ファイルオープン
    for(i=0;i<3;i++)
    {
        if(NULL == (rh_data.fp_in[i] = fopen(argv[3+i], FOPEN_MODE_R)))
        {
            res = ERR_InFile_R + i;
            goto ERR_END;
        }
    }
    for(i=0;i<3;i++)
    {
        if(NULL==(rh_data.fp_out[i]=fopen(argv[6+i], FOPEN_MODE_W)))
        {
            res = ERR_OutFile_H + i;
            goto ERR_END;
        }
    }
    // ファイルサイズチェック
    for(i=0;i<3;i++)
    {
        fseek(rh_data.fp_in[i], 0, SEEK_END);
        if(ftell(rh_data.fp_in[i]) != rh_data.img_size * sizeof(unsigned short) + header_size_rgb_file)
        {
            res = ERR_InFile_Size;
            goto ERR_END;
        }
    }

    /*
     *   メモリ確保
     */
    for(i=0; i<3; i++)
    {
        if(NULL == (rh_data.src[i] = (unsigned short*)malloc(rh_data.img_size*sizeof(unsigned short))))
        {
            res = ERR_MemAlloc;
            goto ERR_END;
        }
        if(NULL == (rh_data.dst[i] = (unsigned short*)malloc(rh_data.img_size*sizeof(unsigned short))))
        {
            res = ERR_MemAlloc;
            goto ERR_END;
        }
    }

    /*
     *   ロード
     */
    load_rgb_data(&rh_data);

    /*
     *   変換
     */
    res = cnv_rgb2hsv(&rh_data);
    if(ERR_Normal != res)
    {
        goto ERR_END;
    }

    /*
     *   セーブ
     */
    res = save_hsv_data(&rh_data);
    if(ERR_Normal != res)
    {
        goto ERR_END;
    }

    /*
     * 終了
     */
    res = ERR_Normal;
ERR_END:;
    // 解放
    for(i=0;i<3;i++)
    {
        if(rh_data.dst[i]) { free(rh_data.dst[i]); rh_data.dst[i] = NULL; }
        if(rh_data.src[i]) { free(rh_data.src[i]); rh_data.src[i] = NULL; }
        if(rh_data.fp_out[i]) { fclose(rh_data.fp_out[i]); rh_data.fp_out[i] = NULL; }
        if(rh_data.fp_in [i]) { fclose(rh_data.fp_in [i]); rh_data.fp_in [i] = NULL; }
    }
    return res;
}

// Load
void load_rgb_data(RGB_HSV_DATA* data)
{
    int             i, j;
    unsigned short  *p, v;

    for(i=0; i<3; i++)
    {
        fseek(data->fp_in[i], header_size_rgb_file, SEEK_SET);
        fread(data->src[i], sizeof(unsigned short), data->img_size, data->fp_in[i]); // 全部読む
        if(0 == g_flagEndian)
        {
            // big endian (入れ替える必要がある)
            p = data->src[i];
            for(j=0; j<data->img_size; j++)
            {
                v = *p;
                *p++ = ((v >> 8) & 0xFF) | ((v << 8) & 0xFF00);
            }
        }
    }
}

// Save
int save_hsv_data(RGB_HSV_DATA* data)
{
    int             i, j;
    unsigned short  *temp = NULL;   // エンディアン変換バッファ
    unsigned short  *p, v;

    // 確保
    if(0 == g_flagEndian)
    {
        // big endian
        if(NULL == (temp = (unsigned short*)malloc(data->img_size * sizeof(unsigned short))))
        {
            return ERR_MemAlloc;
        }
    }
    // 書き込み
    for(i=0; i<3; i++)
    {
        fseek(data->fp_out[i],0,SEEK_SET);
        if(0 == g_flagEndian)
        {
            memcpy(temp, &(data->dst[i][0]), sizeof(unsigned short) * data->img_size);
            p = temp;
            for(j=0; j<data->img_size; j++)
            {
                v = *p;
                *p++ = ((v >> 8) & 0xFF) | ((v << 8) & 0xFF00);
            }
            p = temp;
        } else {
            p = &(data->dst[i][0]);
        }
        fwrite(p, sizeof(unsigned short), data->img_size, data->fp_out[i]);
    }
    // 解放
    if(temp)
    {
        free(temp);
    }
    return 0;
}

// max, min, maxのcolor を調べる
// mxcolor = 'R', 'G', 'B'のいずれか
void get_max_min_mxcolor(/*[in]*/unsigned short R, unsigned short G, unsigned short B, /*[out]*/unsigned short *mx, unsigned short *mn, char *mxcolor)
{
    // max
                  *mx = R; *mxcolor='R';
    if(G > *mx) { *mx = G; *mxcolor='G'; }
    if(B > *mx) { *mx = B; *mxcolor='B'; }
    // min
    *mn = MIN3(R, G, B);
}

// Convert
int cnv_rgb2hsv(RGB_HSV_DATA* data)
{
    unsigned short  max, min;
    char            color_max;  // maxの色 ('R','G','B')
    unsigned short  H,S,V;      // HSV
    double          th;         // H計算用
    double          sub_maxmin; // H計算用 (max - min)
    int             i;          // 汎用

    for(i=0; i<data->img_size; i++)
    {
        // max, min, maxの色を取得
        get_max_min_mxcolor(data->src[0][i], data->src[1][i], data->src[2][i], &max, &min, &color_max);

        // SVを求める
        //  ( http://j...content-available-to-author-only...a.org/wiki/HSV%E8%89%B2%E7%A9%BA%E9%96%93#RGB.E3.81.8B.E3.82.89HSV.E3.81.B8.E3.81.AE.E5.A4.89.E6.8F.9B )
        V = max;
        if(max == min)
        {
            S = 0;
        }else{
            if(max == 0)
            {
                S = 0; // 未定義である
            }else{
                if(comvert_mode == 1)
                {
                    S = (unsigned short)((double)(max - min) / (double)max * 65535.0);
                }
                else if(comvert_mode == 2)
                {
                    S = max - min;
                } else {
                    return ERR_Cnv_mode;
                }
            }
        }
        // Hを求める
        if(max == min)
        {
            th = 0.0; // 未定義である
        }else{
            sub_maxmin = (double)max - (double)min;
            switch(color_max)
            {
            case 'R':th = 60.0 * ((double)data->src[1][i] - (double)data->src[2][i]) / sub_maxmin +   0.0;break;
            case 'G':th = 60.0 * ((double)data->src[2][i] - (double)data->src[0][i]) / sub_maxmin + 120.0;break;
            case 'B':th = 60.0 * ((double)data->src[0][i] - (double)data->src[1][i]) / sub_maxmin + 240.0;break;
            }
        }
        // 0.0以上360.0未満に丸める
        while(th <    0.0)
        {
            th += 360.0;
        }
        while(th >= 360.0)
        {
            th -= 360.0;
        }
        H = (unsigned short)(th / 360.0 * 65535.0); // 360.0の百分率を16bit化
        // HSVをセットする
        data->dst[0][i] = H;
        data->dst[1][i] = S;
        data->dst[2][i] = V;
    }
    return ERR_Normal;
}

// End of  cs160_861.c
