Advertisement
panda_200

YUV2BMP

Apr 5th, 2025
327
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
C++ 6.54 KB | None | 0 0
  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <assert.h>
  4. #include <string.h>
  5. #include "bmp.h"
  6. typedef unsigned char uchar;
  7.  
  8.  
  9. typedef struct {
  10.     int y, u, v;
  11. } YUV;
  12.  
  13. int ch_round(int v) {
  14.     if (v < 0) return 0;
  15.     if (v > 255) return 255;
  16.     return v;
  17. }
  18.  
  19.  
  20. BGR yuv2rgb(YUV yuv) {
  21.     BGR rgb;
  22.     rgb.r = ch_round(yuv.y + 1.402 * (yuv.v - 128));
  23.     rgb.g = ch_round(yuv.y - 0.344136 * (yuv.u - 128) - 0.714136 * (yuv.v - 128));
  24.     rgb.b = ch_round(yuv.y + 1.772 * (yuv.u - 128));
  25.     return rgb;
  26. }
  27.  
  28.  
  29. uchar *read_bytes(FILE *in, int count) {
  30.     uchar *input = malloc(count * sizeof(uchar));
  31.     if (input == NULL) {
  32.         printf("Memory allocation failed\n");
  33.         return NULL;
  34.     }
  35.     size_t read_count = fread(input, sizeof(uchar), count, in);
  36.     if (read_count != count) {
  37.         printf("Read %zu elements, expected %d\n", read_count, count);
  38.         return NULL;
  39.     }
  40.     return input;
  41. }
  42.  
  43. uchar **read_yuv_image(FILE *in, int count, int stride) {
  44.     uchar *luminance = read_bytes(in, count);
  45.     uchar *blue = read_bytes(in, count / (stride * stride));
  46.     uchar *red = read_bytes(in, count / (stride * stride));
  47.  
  48.     if (luminance && blue && red) {
  49.         printf("Data read successfully\n");
  50.     } else {
  51.         free(luminance);
  52.         free(blue);
  53.         free(red);
  54.         return NULL;
  55.     }
  56.     uchar **data = malloc(3 * sizeof(uchar *));
  57.     data[0] = luminance;
  58.     data[1] = blue;
  59.     data[2] = red;
  60.     return data;
  61. }
  62.  
  63. // covert
  64. BGR **transform(uchar *luminance, uchar *blue, uchar *red, int height, int width, int stride,
  65.                 int gray_scale, int up_size, int down_size) {
  66.     int new_height = height * up_size / down_size;
  67.     int new_width = width * up_size / down_size;
  68.     printf("new_height = %d, new_width = %d\n", new_height, new_width);
  69.  
  70.     BGR **data = malloc(height * sizeof(BGR *));
  71.     for (int i = 0; i < height; i++) {
  72.         data[i] = (BGR *) malloc(width * sizeof(BGR));
  73.     }
  74.  
  75.     if (!data) {
  76.         fprintf(stderr, "Memory allocation failed!\n");
  77.         exit(1);
  78.     }
  79.  
  80.  
  81.     for (int i = 0; i < height; i++) {
  82.         for (int j = 0; j < width; j++) {
  83.             if (gray_scale) {
  84.                 BGR rgb = {luminance[i * width + j], luminance[i * width + j], luminance[i * width + j]};
  85.                 data[i][j] = rgb;
  86.             } else {
  87.                 int group_id = (i / stride) * (width / stride) + j / stride;
  88.                 YUV yuv = {luminance[i * width + j], blue[group_id], red[group_id]};
  89.                 BGR rgb = yuv2rgb(yuv);
  90.                 data[i][j] = rgb;
  91.             }
  92.         }
  93.     }
  94.     printf("from yuv to rgb\n");
  95.     // rescale
  96.     BGR **scaled_data = malloc(new_height * sizeof(BGR *));
  97.     for (int i = 0; i < new_height; i++) {
  98.         scaled_data[i] = (BGR *) malloc(new_width * sizeof(BGR));
  99.     }
  100.  
  101.     if (!scaled_data) {
  102.         free(data);
  103.         fprintf(stderr, "Memory allocation failed!\n");
  104.         exit(1);
  105.     }
  106.  
  107.  
  108.     for (int i = 0; i < new_height; i++) {
  109.         for (int j = 0; j < new_width; j++) {
  110.             int old_i = i * down_size / up_size;
  111.             int old_j = j * down_size / up_size;
  112.             if (old_i >= height) {
  113.                 old_i = height - 1;
  114.             }
  115.             if (old_j >= width) {
  116.                 old_j = width - 1;
  117.             }
  118.             scaled_data[i][j] = data[old_i][old_j];
  119.         }
  120.     }
  121.     printf("scaled data\n");
  122.     return scaled_data;
  123. }
  124.  
  125.  
  126. void YUV2BMP(FILE *in, FILE *out, int height, int width, int stride, int gray_scale, int up_size, int down_size) {
  127.     printf("up_size = %d, down_size = %d\n", up_size, down_size);
  128.     int count = height * width;
  129.     uchar **data = read_yuv_image(in, count, stride);
  130.     if (data == NULL) {
  131.         printf("Data read failed\n");
  132.         return;
  133.     }
  134.  
  135.     uchar *luminance = data[0];
  136.     uchar *blue = data[1];
  137.     uchar *red = data[2];
  138.  
  139.     int new_height = height * up_size / down_size;
  140.     int new_width = width * up_size / down_size;
  141.     BMP_Header *bmp_header = malloc(sizeof(BMP_Header));
  142.     fill_bmp_header(bmp_header, new_height, new_width, 24);
  143.     fwrite(bmp_header, sizeof(BMP_Header), 1, out);
  144.  
  145.     BGR **scaled_data = transform(luminance, blue, red, height, width, stride, gray_scale, up_size, down_size);
  146.     printf("data transformed\n");
  147.     for (int i = 0; i < new_height; i++) {
  148.         fwrite(scaled_data[i], sizeof(BGR), new_width, out); // error
  149.         int row_size = (3 * new_width + 3) & ~3;
  150.         fwrite("\0", 1, row_size - 3 * new_width, out); // error
  151.     }
  152.  
  153.     free(data);
  154.     free(scaled_data);
  155. }
  156.  
  157.  
  158. char *is_argkey(char *arg) {
  159.     if (strlen(arg) <= 2) {
  160.         return NULL;
  161.     }
  162.     if (arg[0] == '-' && arg[1] == '-') {
  163.         return arg + 2;
  164.     }
  165.     return NULL;
  166. }
  167.  
  168. int main(int argc, char *argv[]) {
  169.     // printf("amount of arguments %d \n", argc);
  170.     char *input_file_name;
  171.     char *output_file_name;
  172.     int width;
  173.     int height;
  174.     int is_gray_scale = 0; // 1 - if grayscale, 0 - if not
  175.     int up_size = 1, down_size = 1;
  176.     for (int i = 1; i < argc; i++) {
  177.         char *key = is_argkey(argv[i]);
  178.         if (key == NULL) {
  179.             printf("No valid argument for command line %s \n", argv[i]);
  180.             return 1;
  181.         }
  182.         if (strcmp(key, "input") == 0) {
  183.             input_file_name = argv[++i];
  184.         } else if (strcmp(key, "output") == 0) { // ++i есть?
  185.             output_file_name = argv[++i];
  186.         } else if (strcmp(key, "width") == 0) {
  187.             width = atoi(argv[++i]);
  188.         } else if (strcmp(key, "height") == 0) {
  189.             height = atoi(argv[++i]);
  190.         } else if (strcmp(key, "to–gray–scale") == 0) {
  191.             is_gray_scale = 1;
  192.         } else if (strcmp(key, "up-size") == 0) {
  193.             up_size = atoi(argv[++i]);
  194.         } else if (strcmp(key, "down-size") == 0) {
  195.             down_size = atoi(argv[++i]);
  196.         } else {
  197.             printf("key is not find %s \n", key);
  198.             return 1;
  199.         }
  200.     }
  201.  
  202.     if (width <= 0 || height <= 0 || up_size <= 0 || down_size <= 0) {
  203.         perror("Can't get minus arguments.\n");
  204.         return 1;
  205.     }
  206.  
  207.     FILE *in = fopen(input_file_name, "rb");
  208.     if (!in) {
  209.         perror("Error opening input file");
  210.         return 1;
  211.     }
  212.  
  213.     FILE *out = fopen(output_file_name, "wb");
  214.     if (!out) {
  215.         perror("Error opening output file");
  216.         fclose(in);
  217.         return 1;
  218.     }
  219.  
  220.  
  221.     YUV2BMP(in, out, height, width, 2, is_gray_scale, up_size, down_size);
  222.  
  223.     fclose(in);
  224.     fclose(out);
  225. }
  226.  
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement