আমি সম্প্রতি সি তে এই প্রকল্পটি হাতে নিয়েছি নীচের কোডটি নিম্নলিখিতটি করে:
1) চিত্রের বর্তমান ওরিয়েন্টেশন পান।
2) ফাঁকা রেখে APP1
(এক্সিফ ডেটা) এবং APP2
(ফ্ল্যাশপিক্স ডেটা) থাকা সমস্ত ডেটা সরিয়ে দেয়।
3) APP1
ওরিয়েন্টেশন চিহ্নিতকরণটি পুনরুদ্ধার করে এবং এটি মূল মানকে সেট করে।
4) প্রথম EOI
চিহ্নিতকারী (চিত্রের শেষ ) সন্ধান করে এবং নেসেসারি হলে ফাইলটি কেটে দেয়।
প্রথমে কিছু বিষয় লক্ষণীয়:
1) এই প্রোগ্রামটি আমার নিকন ক্যামেরার জন্য ব্যবহৃত হয়। নিকনের জেপিজি ফর্ম্যাটটি এটি তৈরি করা প্রতিটি ফাইলের একেবারে শেষের দিকে কিছুটা যুক্ত করে। তারা দ্বিতীয় EOI
মার্কার তৈরি করে চিত্রের শেষের দিকে এই ডেটাটি এনকোড করে । সাধারণত চিত্রের প্রোগ্রামগুলি প্রথম EOI
চিহ্নিত পাওয়া যায় read নিকনের কাছে এর পরে তথ্য রয়েছে যা আমার প্রোগ্রামটি কেটে দেয়।
2) এটি নিকন ফর্ম্যাটের জন্য, এটি big endian
বাইট ক্রম ধরে । যদি আপনার চিত্র ফাইলটি ব্যবহার করে তবে little endian
কিছু সমন্বয় করা দরকার।
3) ImageMagick
এক্সিফ ডেটা ফালা ব্যবহার করার চেষ্টা করার সময় , আমি লক্ষ্য করেছি যে আমি যেটি শুরু করেছি তার চেয়ে বড় ফাইল দিয়ে শেষ করেছি। এটি আমাকে বিশ্বাস করতে পরিচালিত করে যে Imagemagick
আপনি যে তথ্যটি সরিয়ে নিতে চান সেটি এনকোডিং করছে এবং এটি ফাইলের অন্য কোথাও সংরক্ষণ করছে। আমাকে পুরাতন ফ্যাশন বলুন, কিন্তু আমি যখন কোনও ফাইল থেকে কিছু সরিয়ে ফেলি, আমি একই আকারের না হলে একটি ফাইলের আকার ছোট হতে চাই। অন্য কোনও ফলাফল ডেটা মাইনিংয়ের পরামর্শ দেয়।
এবং এখানে কোড:
#include <stdio.h>
#include <stdlib.h>
#include <libgen.h>
#include <string.h>
#include <errno.h>
// Declare constants.
#define COMMAND_SIZE 500
#define RETURN_SUCCESS 1
#define RETURN_FAILURE 0
#define WORD_SIZE 15
int check_file_jpg (void);
int check_file_path (char *file);
int get_marker (void);
char * ltoa (long num);
void process_image (char *file);
// Declare global variables.
FILE *fp;
int orientation;
char *program_name;
int main (int argc, char *argv[])
{
// Set program name for error reporting.
program_name = basename(argv[0]);
// Check for at least one argument.
if(argc < 2)
{
fprintf(stderr, "usage: %s IMAGE_FILE...\n", program_name);
exit(EXIT_FAILURE);
}
// Process all arguments.
for(int x = 1; x < argc; x++)
process_image(argv[x]);
exit(EXIT_SUCCESS);
}
void process_image (char *file)
{
char command[COMMAND_SIZE + 1];
// Check that file exists.
if(check_file_path(file) == RETURN_FAILURE)
return;
// Check that file is an actual JPEG file.
if(check_file_jpg() == RETURN_FAILURE)
{
fclose(fp);
return;
}
// Jump to orientation marker and store value.
fseek(fp, 55, SEEK_SET);
orientation = fgetc(fp);
// Recreate the APP1 marker with just the orientation tag listed.
fseek(fp, 21, SEEK_SET);
fputc(1, fp);
fputc(1, fp);
fputc(18, fp);
fputc(0, fp);
fputc(3, fp);
fputc(0, fp);
fputc(0, fp);
fputc(0, fp);
fputc(1, fp);
fputc(0, fp);
fputc(orientation, fp);
// Blank the rest of the APP1 marker with '\0'.
for(int x = 0; x < 65506; x++)
fputc(0, fp);
// Blank the second APP1 marker with '\0'.
fseek(fp, 4, SEEK_CUR);
for(int x = 0; x < 2044; x++)
fputc(0, fp);
// Blank the APP2 marker with '\0'.
fseek(fp, 4, SEEK_CUR);
for(int x = 0; x < 4092; x++)
fputc(0, fp);
// Jump the the SOS marker.
fseek(fp, 72255, SEEK_SET);
while(1)
{
// Truncate the file once the first EOI marker is found.
if(fgetc(fp) == 255 && fgetc(fp) == 217)
{
strcpy(command, "truncate -s ");
strcat(command, ltoa(ftell(fp)));
strcat(command, " ");
strcat(command, file);
fclose(fp);
system(command);
break;
}
}
}
int get_marker (void)
{
int c;
// Check to make sure marker starts with 0xFF.
if((c = fgetc(fp)) != 0xFF)
{
fprintf(stderr, "%s: get_marker: invalid marker start (should be FF, is %2X)\n", program_name, c);
return(RETURN_FAILURE);
}
// Return the next character.
return(fgetc(fp));
}
int check_file_jpg (void)
{
// Check if marker is 0xD8.
if(get_marker() != 0xD8)
{
fprintf(stderr, "%s: check_file_jpg: not a valid jpeg image\n", program_name);
return(RETURN_FAILURE);
}
return(RETURN_SUCCESS);
}
int check_file_path (char *file)
{
// Open file.
if((fp = fopen(file, "rb+")) == NULL)
{
fprintf(stderr, "%s: check_file_path: fopen failed (%s) (%s)\n", program_name, strerror(errno), file);
return(RETURN_FAILURE);
}
return(RETURN_SUCCESS);
}
char * ltoa (long num)
{
// Declare variables.
int ret;
int x = 1;
int y = 0;
static char temp[WORD_SIZE + 1];
static char word[WORD_SIZE + 1];
// Stop buffer overflow.
temp[0] = '\0';
// Keep processing until value is zero.
while(num > 0)
{
ret = num % 10;
temp[x++] = 48 + ret;
num /= 10;
}
// Reverse the word.
while(y < x)
{
word[y] = temp[x - y - 1];
y++;
}
return word;
}
আশা করি এটি কাউকে সাহায্য করবে!