Having trouble running Permutations in Do-while loop

I'm still pretty new at programming, but I know enough to piece together other people's code usually. I'm currently stuck at trying to get a string to run a permutation, check it against another sting, and if it's different then move on to the next permutation. This is all happening with SHA256 encryption going on too.

The way I want it to work is I put a the seed/hashed string in the txt file "seed.txt" and that gets pulled into the main function as string desired. Then string s, which will be the unhashed string version of string desired, wiil run permutations until it becomes the correct sequence of characters that will encrypt to be the same string as desired.

The code works when I have the string s (in this case "Hello") already in order, but when I switch the order around at all ex) "elHlo" it won't run permutations until that eventually gets to the permutation "Hello" which would end the program. Any help is appreciated.

Main code
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
#include <cstring>
#include <fstream>
#include "sha256.h"
#include <iostream>
#include <algorithm>

using namespace std;

int main(int argc, char *argv[])
{
  string desired = "a";
  ifstream myfile;
  myfile.open ("seed.txt");
  myfile >> desired;

  string s = "Hello";

  string working = "dcba";

  do
    {   
    string working = s;
    string sha = sha256(working);
      
    if (sha != desired)
      continue;

    else if (sha == desired)
      {cout << s << " : " << sha << " \n" << "Desired: " << desired; 
      return 0;
      }

    /*else      program just stopped running, wouldnt print. should still be fine
      {
        cout << "No Match \n";
        return 0;
      }*/
    }
   while (next_permutation(s.begin(), s.end()));

   myfile.close();

}

const unsigned int SHA256::sha256_k[64] = //UL = uint32
            {0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5,
             0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5,
             0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3,
             0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174,
             0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc,
             0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da,
             0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7,
             0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967,
             0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13,
             0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85,
             0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3,
             0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070,
             0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5,
             0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3,
             0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208,
             0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2};
 
void SHA256::transform(const unsigned char *message, unsigned int block_nb)
{
    uint32 w[64];
    uint32 wv[8];
    uint32 t1, t2;
    const unsigned char *sub_block;
    int i;
    int j;
    for (i = 0; i < (int) block_nb; i++) {
        sub_block = message + (i << 6);
        for (j = 0; j < 16; j++) {
            SHA2_PACK32(&sub_block[j << 2], &w[j]);
        }
        for (j = 16; j < 64; j++) {
            w[j] =  SHA256_F4(w[j -  2]) + w[j -  7] + SHA256_F3(w[j - 15]) + w[j - 16];
        }
        for (j = 0; j < 8; j++) {
            wv[j] = m_h[j];
        }
        for (j = 0; j < 64; j++) {
            t1 = wv[7] + SHA256_F2(wv[4]) + SHA2_CH(wv[4], wv[5], wv[6])
                + sha256_k[j] + w[j];
            t2 = SHA256_F1(wv[0]) + SHA2_MAJ(wv[0], wv[1], wv[2]);
            wv[7] = wv[6];
            wv[6] = wv[5];
            wv[5] = wv[4];
            wv[4] = wv[3] + t1;
            wv[3] = wv[2];
            wv[2] = wv[1];
            wv[1] = wv[0];
            wv[0] = t1 + t2;
        }
        for (j = 0; j < 8; j++) {
            m_h[j] += wv[j];
        }
    }
}
 
void SHA256::init()
{
    m_h[0] = 0x6a09e667;
    m_h[1] = 0xbb67ae85;
    m_h[2] = 0x3c6ef372;
    m_h[3] = 0xa54ff53a;
    m_h[4] = 0x510e527f;
    m_h[5] = 0x9b05688c;
    m_h[6] = 0x1f83d9ab;
    m_h[7] = 0x5be0cd19;
    m_len = 0;
    m_tot_len = 0;
}
 
void SHA256::update(const unsigned char *message, unsigned int len)
{
    unsigned int block_nb;
    unsigned int new_len, rem_len, tmp_len;
    const unsigned char *shifted_message;
    tmp_len = SHA224_256_BLOCK_SIZE - m_len;
    rem_len = len < tmp_len ? len : tmp_len;
    memcpy(&m_block[m_len], message, rem_len);
    if (m_len + len < SHA224_256_BLOCK_SIZE) {
        m_len += len;
        return;
    }
    new_len = len - rem_len;
    block_nb = new_len / SHA224_256_BLOCK_SIZE;
    shifted_message = message + rem_len;
    transform(m_block, 1);
    transform(shifted_message, block_nb);
    rem_len = new_len % SHA224_256_BLOCK_SIZE;
    memcpy(m_block, &shifted_message[block_nb << 6], rem_len);
    m_len = rem_len;
    m_tot_len += (block_nb + 1) << 6;
}
 
void SHA256::final(unsigned char *digest)
{
    unsigned int block_nb;
    unsigned int pm_len;
    unsigned int len_b;
    int i;
    block_nb = (1 + ((SHA224_256_BLOCK_SIZE - 9)
                     < (m_len % SHA224_256_BLOCK_SIZE)));
    len_b = (m_tot_len + m_len) << 3;
    pm_len = block_nb << 6;
    memset(m_block + m_len, 0, pm_len - m_len);
    m_block[m_len] = 0x80;
    SHA2_UNPACK32(len_b, m_block + pm_len - 4);
    transform(m_block, block_nb);
    for (i = 0 ; i < 8; i++) {
        SHA2_UNPACK32(m_h[i], &digest[i << 2]);
    }
}
 
string sha256(string input)
{
    unsigned char digest[SHA256::DIGEST_SIZE];
    memset(digest,0,SHA256::DIGEST_SIZE);
 
    SHA256 ctx = SHA256();
    ctx.init();
    ctx.update( (unsigned char*)input.c_str(), input.length());
    ctx.final(digest);
 
    char buf[2*SHA256::DIGEST_SIZE+1];
    buf[2*SHA256::DIGEST_SIZE] = 0;
    for (int i = 0; i < SHA256::DIGEST_SIZE; i++)
        sprintf(buf+i*2, "%02x", digest[i]);
    return string(buf);
}


sha256.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
#ifndef SHA256_H
#define SHA256_H
#include <string>
 
class SHA256
{
protected:
    typedef unsigned char uint8;
    typedef unsigned int uint32;
    typedef unsigned long long uint64;
 
    const static uint32 sha256_k[];
    static const unsigned int SHA224_256_BLOCK_SIZE = (512/8);
public:
    void init();
    void update(const unsigned char *message, unsigned int len);
    void final(unsigned char *digest);
    static const unsigned int DIGEST_SIZE = ( 256 / 8);
 
protected:
    void transform(const unsigned char *message, unsigned int block_nb);
    unsigned int m_tot_len;
    unsigned int m_len;
    unsigned char m_block[2*SHA224_256_BLOCK_SIZE];
    uint32 m_h[8];
};
 
std::string sha256(std::string input);
 
#define SHA2_SHFR(x, n)    (x >> n)
#define SHA2_ROTR(x, n)   ((x >> n) | (x << ((sizeof(x) << 3) - n)))
#define SHA2_ROTL(x, n)   ((x << n) | (x >> ((sizeof(x) << 3) - n)))
#define SHA2_CH(x, y, z)  ((x & y) ^ (~x & z))
#define SHA2_MAJ(x, y, z) ((x & y) ^ (x & z) ^ (y & z))
#define SHA256_F1(x) (SHA2_ROTR(x,  2) ^ SHA2_ROTR(x, 13) ^ SHA2_ROTR(x, 22))
#define SHA256_F2(x) (SHA2_ROTR(x,  6) ^ SHA2_ROTR(x, 11) ^ SHA2_ROTR(x, 25))
#define SHA256_F3(x) (SHA2_ROTR(x,  7) ^ SHA2_ROTR(x, 18) ^ SHA2_SHFR(x,  3))
#define SHA256_F4(x) (SHA2_ROTR(x, 17) ^ SHA2_ROTR(x, 19) ^ SHA2_SHFR(x, 10))
#define SHA2_UNPACK32(x, str)                 \
{                                             \
    *((str) + 3) = (uint8) ((x)      );       \
    *((str) + 2) = (uint8) ((x) >>  8);       \
    *((str) + 1) = (uint8) ((x) >> 16);       \
    *((str) + 0) = (uint8) ((x) >> 24);       \
}
#define SHA2_PACK32(str, x)                   \
{                                             \
    *(x) =   ((uint32) *((str) + 3)      )    \
           | ((uint32) *((str) + 2) <<  8)    \
           | ((uint32) *((str) + 1) << 16)    \
           | ((uint32) *((str) + 0) << 24);   \
}
#endif


seed.txt
 
185f8db32271fe25f561a6fc938b2e264306ec304eda518007d1764826381969
Last edited on
In order to use next_permutation to go through all the permutations, you need to start with a sorted string. So you need to call sort(s.begin(), s.end()) before your loop. In the case of "Hello", it's already in order, so it doesn't really test the permutation part of your code.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
int main() {
    ifstream myfile("seed.txt");
    string desired;
    myfile >> desired;
    myfile.close();

    string s = "leHol";
    sort(s.begin(), s.end());

    do {
        string sha = sha256(s);
        if (sha == desired) {
            cout << s << " : " << sha << '\n';
            break;
        }
    } while (next_permutation(s.begin(), s.end()));
}

Thank you, that worked. I've noticed that the code runs a little slowly when I have larger strings. Would that be the sorting part of the code? In which case if I have string s already sorted it would save a lot of time. If it is the permutation part of the code I don't think there is a way to speed it up.
It's the permutations that are the problem. The number of permutations is the factorial of the string length. So if the string is 10 letters long, there are 10! = 3,628,800 permutations.

For 15 letters it's 15! = 1,307,674,368,000.
At one hundred million perms per second (if you could do it that fast) it would take over 3.6 hours.

20! = 2,432,902,008,176,640,000
Processing a billion perms per second it would take over 77 years.

Compare the time for sorting to the time to go through the perms of a string of size 10:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
#include <algorithm>
#include <chrono>
#include <iostream>
#include <string>

template<typename T>
inline void show_time(T start, T end) {
    std::cout << std::chrono::duration_cast<std::chrono::microseconds>
                 (end - start).count() << " microseconds\n";
}

int main() {
    auto clk = std::chrono::high_resolution_clock{};
    std::string s{"8360297514"};

    std::cout << "len: " << s.size() << '\n';

    auto start = clk.now();
    std::sort(s.begin(), s.end());
    show_time(start, clk.now());

    start = clk.now();
    int cnt{0};
    do {
        if (int(s[0] - '0') % 2 == 0) // count perms that start with even digit
            ++cnt;
    } while (std::next_permutation(s.begin(), s.end()));
    show_time(start, clk.now());

    std::cout << "cnt: " << cnt << '\n';
}

Last edited on
Wow. I knew perms would take a while, but that's a real long time. Do the perms do repeating characters too? Like for 123 does it do: 123, 132, 213, 231, 321, 312. or is it like: 111, 112, 113, 121... ?
Last edited on
Perms don't repeat chars, unless the input string has repeating chars. So 123, being 3 long, will have 3! = 3*2*1 = 6 permutations: 123, 132, 213, 231, 312, 321.
Topic archived. No new replies allowed.